* main: (213 commits) website/docs: configuration: fix typo in kubectl command (#10492) website/integrations: fix typo in minio instructions (#10500) web: bump @typescript-eslint/eslint-plugin from 7.5.0 to 7.16.0 in /tests/wdio (#10496) website: bump prettier from 3.3.2 to 3.3.3 in /website (#10493) core: bump ruff from 0.5.1 to 0.5.2 (#10494) web: bump @typescript-eslint/parser from 7.5.0 to 7.16.0 in /tests/wdio (#10495) web: bump eslint-plugin-sonarjs from 0.25.1 to 1.0.3 in /tests/wdio (#10498) web: bump prettier from 3.3.2 to 3.3.3 in /tests/wdio (#10497) web: bump pseudolocale from 2.0.0 to 2.1.0 in /web (#10499) core: bump goauthentik.io/api/v3 from 3.2024061.1 to 3.2024061.2 (#10491) web: bump API Client version (#10488) flows: remove stage challenge type (#10476) core: bump github.com/redis/go-redis/v9 from 9.5.3 to 9.5.4 (#10469) core: bump goauthentik.io/api/v3 from 3.2024060.6 to 3.2024061.1 (#10470) web: bump the babel group across 1 directory with 2 updates (#10471) web: bump the storybook group across 1 directory with 7 updates (#10472) core: bump coverage from 7.5.4 to 7.6.0 (#10473) website/docs: air gapped: clarify .env usage at the top for Kubernetes installations (#10447) website/docs: air gapped: update "see configuration" wording (#10448) website/docs: Add Kubernetes Bootstrap Instructions (#9541) ...
268 lines
10 KiB
TypeScript
268 lines
10 KiB
TypeScript
import { DEFAULT_CONFIG } from "@goauthentik/common/api/config";
|
|
import { docLink } from "@goauthentik/common/global";
|
|
import { groupBy } from "@goauthentik/common/utils";
|
|
import "@goauthentik/elements/CodeMirror";
|
|
import { CodeMirrorMode } from "@goauthentik/elements/CodeMirror";
|
|
import "@goauthentik/elements/ak-dual-select/ak-dual-select-provider";
|
|
import { DataProvider, DualSelectPair } from "@goauthentik/elements/ak-dual-select/types";
|
|
import "@goauthentik/elements/forms/FormGroup";
|
|
import "@goauthentik/elements/forms/HorizontalFormElement";
|
|
import { ModelForm } from "@goauthentik/elements/forms/ModelForm";
|
|
import "@goauthentik/elements/forms/SearchSelect";
|
|
import { PaginatedResponse } from "@goauthentik/elements/table/Table";
|
|
import YAML from "yaml";
|
|
|
|
import { msg } from "@lit/localize";
|
|
import { TemplateResult, html } from "lit";
|
|
import { customElement, property, state } from "lit/decorators.js";
|
|
import { ifDefined } from "lit/directives/if-defined.js";
|
|
import { map } from "lit/directives/map.js";
|
|
|
|
import {
|
|
Outpost,
|
|
OutpostDefaultConfig,
|
|
OutpostTypeEnum,
|
|
OutpostsApi,
|
|
OutpostsServiceConnectionsAllListRequest,
|
|
ProvidersApi,
|
|
ServiceConnection,
|
|
} from "@goauthentik/api";
|
|
|
|
interface ProviderBase {
|
|
pk: number;
|
|
name: string;
|
|
assignedBackchannelApplicationName?: string;
|
|
assignedApplicationName?: string;
|
|
}
|
|
|
|
const api = () => new ProvidersApi(DEFAULT_CONFIG);
|
|
const providerListArgs = (page: number, search = "") => ({
|
|
ordering: "name",
|
|
applicationIsnull: false,
|
|
pageSize: 20,
|
|
search: search.trim(),
|
|
page,
|
|
});
|
|
|
|
const dualSelectPairMaker = (item: ProviderBase): DualSelectPair => {
|
|
const label = item.assignedBackchannelApplicationName
|
|
? item.assignedBackchannelApplicationName
|
|
: item.assignedApplicationName;
|
|
return [
|
|
`${item.pk}`,
|
|
html`<div class="selection-main">${label}</div>
|
|
<div class="selection-desc">${item.name}</div>`,
|
|
label,
|
|
];
|
|
};
|
|
|
|
const provisionMaker = (results: PaginatedResponse<ProviderBase>) => ({
|
|
pagination: results.pagination,
|
|
options: results.results.map(dualSelectPairMaker),
|
|
});
|
|
|
|
const proxyListFetch = async (page: number, search = "") =>
|
|
provisionMaker(await api().providersProxyList(providerListArgs(page, search)));
|
|
|
|
const ldapListFetch = async (page: number, search = "") =>
|
|
provisionMaker(await api().providersLdapList(providerListArgs(page, search)));
|
|
|
|
const radiusListFetch = async (page: number, search = "") =>
|
|
provisionMaker(await api().providersRadiusList(providerListArgs(page, search)));
|
|
|
|
const racListProvider = async (page: number, search = "") =>
|
|
provisionMaker(await api().providersRacList(providerListArgs(page, search)));
|
|
|
|
function providerProvider(type: OutpostTypeEnum): DataProvider {
|
|
switch (type) {
|
|
case OutpostTypeEnum.Proxy:
|
|
return proxyListFetch;
|
|
case OutpostTypeEnum.Ldap:
|
|
return ldapListFetch;
|
|
case OutpostTypeEnum.Radius:
|
|
return radiusListFetch;
|
|
case OutpostTypeEnum.Rac:
|
|
return racListProvider;
|
|
default:
|
|
throw new Error(`Unrecognized OutputType: ${type}`);
|
|
}
|
|
}
|
|
|
|
@customElement("ak-outpost-form")
|
|
export class OutpostForm extends ModelForm<Outpost, string> {
|
|
@property()
|
|
type: OutpostTypeEnum = OutpostTypeEnum.Proxy;
|
|
|
|
@property({ type: Boolean })
|
|
embedded = false;
|
|
|
|
@state()
|
|
providers?: DataProvider;
|
|
defaultConfig?: OutpostDefaultConfig;
|
|
|
|
async loadInstance(pk: string): Promise<Outpost> {
|
|
const o = await new OutpostsApi(DEFAULT_CONFIG).outpostsInstancesRetrieve({
|
|
uuid: pk,
|
|
});
|
|
this.type = o.type || OutpostTypeEnum.Proxy;
|
|
return o;
|
|
}
|
|
|
|
async load(): Promise<void> {
|
|
this.defaultConfig = await new OutpostsApi(
|
|
DEFAULT_CONFIG,
|
|
).outpostsInstancesDefaultSettingsRetrieve();
|
|
this.providers = providerProvider(this.type);
|
|
}
|
|
|
|
getSuccessMessage(): string {
|
|
return this.instance
|
|
? msg("Successfully updated outpost.")
|
|
: msg("Successfully created outpost.");
|
|
}
|
|
|
|
async send(data: Outpost): Promise<Outpost> {
|
|
if (this.instance) {
|
|
return new OutpostsApi(DEFAULT_CONFIG).outpostsInstancesUpdate({
|
|
uuid: this.instance.pk || "",
|
|
outpostRequest: data,
|
|
});
|
|
} else {
|
|
return new OutpostsApi(DEFAULT_CONFIG).outpostsInstancesCreate({
|
|
outpostRequest: data,
|
|
});
|
|
}
|
|
}
|
|
|
|
renderForm(): TemplateResult {
|
|
const typeOptions = [
|
|
[OutpostTypeEnum.Proxy, msg("Proxy")],
|
|
[OutpostTypeEnum.Ldap, msg("LDAP")],
|
|
[OutpostTypeEnum.Radius, msg("Radius")],
|
|
[OutpostTypeEnum.Rac, msg("RAC")],
|
|
];
|
|
|
|
return html` <ak-form-element-horizontal label=${msg("Name")} ?required=${true} name="name">
|
|
<input
|
|
type="text"
|
|
value="${ifDefined(this.instance?.name)}"
|
|
class="pf-c-form-control"
|
|
required
|
|
/>
|
|
</ak-form-element-horizontal>
|
|
<ak-form-element-horizontal label=${msg("Type")} ?required=${true} name="type">
|
|
<select
|
|
class="pf-c-form-control"
|
|
@change=${(ev: Event) => {
|
|
const target = ev.target as HTMLSelectElement;
|
|
this.type = target.selectedOptions[0].value as OutpostTypeEnum;
|
|
this.load();
|
|
}}
|
|
>
|
|
${map(
|
|
typeOptions,
|
|
([instanceType, label]) =>
|
|
html` <option
|
|
value=${instanceType}
|
|
?selected=${this.instance?.type === instanceType}
|
|
>
|
|
${label}
|
|
</option>`,
|
|
)}
|
|
</select>
|
|
</ak-form-element-horizontal>
|
|
<ak-form-element-horizontal label=${msg("Integration")} name="serviceConnection">
|
|
<ak-search-select
|
|
.fetchObjects=${async (query?: string): Promise<ServiceConnection[]> => {
|
|
const args: OutpostsServiceConnectionsAllListRequest = {
|
|
ordering: "name",
|
|
};
|
|
if (query !== undefined) {
|
|
args.search = query;
|
|
}
|
|
const items = await new OutpostsApi(
|
|
DEFAULT_CONFIG,
|
|
).outpostsServiceConnectionsAllList(args);
|
|
return items.results;
|
|
}}
|
|
.renderElement=${(item: ServiceConnection): string => {
|
|
return item.name;
|
|
}}
|
|
.value=${(item: ServiceConnection | undefined): string | undefined => {
|
|
return item?.pk;
|
|
}}
|
|
.groupBy=${(items: ServiceConnection[]) => {
|
|
return groupBy(items, (item) => item.verboseName);
|
|
}}
|
|
.selected=${(item: ServiceConnection, items: ServiceConnection[]): boolean => {
|
|
let selected = this.instance?.serviceConnection === item.pk;
|
|
if (items.length === 1 && !this.instance) {
|
|
selected = true;
|
|
}
|
|
return selected;
|
|
}}
|
|
?blankable=${true}
|
|
>
|
|
</ak-search-select>
|
|
<p class="pf-c-form__helper-text">
|
|
${msg(
|
|
"Selecting an integration enables the management of the outpost by authentik.",
|
|
)}
|
|
</p>
|
|
<p class="pf-c-form__helper-text">
|
|
<a
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
href="${docLink("/docs/outposts?utm_source=authentik")}"
|
|
>${msg("See documentation")}</a
|
|
>.
|
|
</p>
|
|
</ak-form-element-horizontal>
|
|
<ak-form-element-horizontal
|
|
label=${msg("Applications")}
|
|
?required=${!this.embedded}
|
|
name="providers"
|
|
>
|
|
<ak-dual-select-provider
|
|
.provider=${this.providers}
|
|
.selected=${(this.instance?.providersObj ?? []).map(dualSelectPairMaker)}
|
|
available-label="${msg("Available Applications")}"
|
|
selected-label="${msg("Selected Applications")}"
|
|
></ak-dual-select-provider>
|
|
</ak-form-element-horizontal>
|
|
<ak-form-group aria-label=${msg("Advanced settings")}>
|
|
<span slot="header"> ${msg("Advanced settings")} </span>
|
|
<div slot="body" class="pf-c-form">
|
|
<ak-form-element-horizontal label=${msg("Configuration")} name="config">
|
|
<ak-codemirror
|
|
mode=${CodeMirrorMode.YAML}
|
|
value="${YAML.stringify(
|
|
this.instance ? this.instance.config : this.defaultConfig?.config,
|
|
)}"
|
|
></ak-codemirror>
|
|
<p class="pf-c-form__helper-text">
|
|
${msg("Set custom attributes using YAML or JSON.")}
|
|
</p>
|
|
<p class="pf-c-form__helper-text">
|
|
${msg("See more here:")}
|
|
<a
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
href="${docLink(
|
|
"/docs/outposts?utm_source=authentik#configuration",
|
|
)}"
|
|
>${msg("Documentation")}</a
|
|
>
|
|
</p>
|
|
</ak-form-element-horizontal>
|
|
</div>
|
|
</ak-form-group>`;
|
|
}
|
|
}
|
|
|
|
declare global {
|
|
interface HTMLElementTagNameMap {
|
|
"ak-outpost-form": OutpostForm;
|
|
}
|
|
}
|