263 lines
8.1 KiB
TypeScript
263 lines
8.1 KiB
TypeScript
|
import React, { useState } from "react";
|
||
|
import { useTranslation } from "react-i18next";
|
||
|
import {
|
||
|
AlertVariant,
|
||
|
Button,
|
||
|
ButtonVariant,
|
||
|
Label,
|
||
|
PageSection,
|
||
|
Text,
|
||
|
TextContent,
|
||
|
} from "@patternfly/react-core";
|
||
|
import { FormPanel } from "../components/scroll-form/FormPanel";
|
||
|
import { KeycloakDataTable } from "../components/table-toolbar/KeycloakDataTable";
|
||
|
import { cellWidth } from "@patternfly/react-table";
|
||
|
import { Link, useParams } from "react-router-dom";
|
||
|
import { useAdminClient } from "../context/auth/AdminClient";
|
||
|
import { emptyFormatter, upperCaseFormatter } from "../util";
|
||
|
import type IdentityProviderRepresentation from "@keycloak/keycloak-admin-client/lib/defs/identityProviderRepresentation";
|
||
|
import type FederatedIdentityRepresentation from "@keycloak/keycloak-admin-client/lib/defs/federatedIdentityRepresentation";
|
||
|
import { useRealm } from "../context/realm-context/RealmContext";
|
||
|
import { useServerInfo } from "../context/server-info/ServerInfoProvider";
|
||
|
import _ from "lodash";
|
||
|
import { useConfirmDialog } from "../components/confirm-dialog/ConfirmDialog";
|
||
|
import { useAlerts } from "../components/alert/Alerts";
|
||
|
import { UserIdpModal } from "./UserIdPModal";
|
||
|
import { toIdentityProviderTab } from "../identity-providers/routes/IdentityProviderTab";
|
||
|
|
||
|
export const UserIdentityProviderLinks = () => {
|
||
|
const [key, setKey] = useState(0);
|
||
|
const [federatedId, setFederatedId] = useState("");
|
||
|
const [isLinkIdPModalOpen, setIsLinkIdPModalOpen] = useState(false);
|
||
|
|
||
|
const adminClient = useAdminClient();
|
||
|
const { id } = useParams<{ id: string }>();
|
||
|
const { realm } = useRealm();
|
||
|
const { addAlert, addError } = useAlerts();
|
||
|
const { t } = useTranslation("users");
|
||
|
|
||
|
const refresh = () => setKey(new Date().getTime());
|
||
|
|
||
|
const handleModalToggle = () => {
|
||
|
setIsLinkIdPModalOpen(!isLinkIdPModalOpen);
|
||
|
};
|
||
|
|
||
|
const identityProviders = useServerInfo().identityProviders;
|
||
|
|
||
|
const getFederatedIdentities = async () => {
|
||
|
return await adminClient.users.listFederatedIdentities({ id });
|
||
|
};
|
||
|
|
||
|
const getAvailableIdPs = async () => {
|
||
|
return (await adminClient.realms.findOne({ realm })).identityProviders;
|
||
|
};
|
||
|
|
||
|
const linkedIdPsLoader = async () => {
|
||
|
return getFederatedIdentities();
|
||
|
};
|
||
|
|
||
|
const availableIdPsLoader = async () => {
|
||
|
const linkedNames = (await getFederatedIdentities()).map(
|
||
|
(x) => x.identityProvider
|
||
|
);
|
||
|
|
||
|
return (await getAvailableIdPs())?.filter(
|
||
|
(item) => !linkedNames.includes(item.alias)
|
||
|
)!;
|
||
|
};
|
||
|
|
||
|
const [toggleUnlinkDialog, UnlinkConfirm] = useConfirmDialog({
|
||
|
titleKey: t("users:unlinkAccountTitle", {
|
||
|
provider: _.capitalize(federatedId),
|
||
|
}),
|
||
|
messageKey: t("users:unlinkAccountConfirm", {
|
||
|
provider: _.capitalize(federatedId),
|
||
|
}),
|
||
|
continueButtonLabel: "users:unlink",
|
||
|
continueButtonVariant: ButtonVariant.primary,
|
||
|
onConfirm: async () => {
|
||
|
try {
|
||
|
await adminClient.users.delFromFederatedIdentity({
|
||
|
id,
|
||
|
federatedIdentityId: federatedId,
|
||
|
});
|
||
|
addAlert(t("common:mappingDeletedSuccess"), AlertVariant.success);
|
||
|
refresh();
|
||
|
} catch (error) {
|
||
|
addError("common:mappingDeletedError", error);
|
||
|
}
|
||
|
},
|
||
|
});
|
||
|
|
||
|
const idpLinkRenderer = (idp: FederatedIdentityRepresentation) => {
|
||
|
return (
|
||
|
<Link
|
||
|
to={toIdentityProviderTab({
|
||
|
realm,
|
||
|
id: idp.identityProvider!,
|
||
|
tab: "settings",
|
||
|
})}
|
||
|
>
|
||
|
{_.capitalize(idp.identityProvider)}
|
||
|
</Link>
|
||
|
);
|
||
|
};
|
||
|
|
||
|
const badgeRenderer1 = (idp: FederatedIdentityRepresentation) => {
|
||
|
const groupName = identityProviders?.find(
|
||
|
(provider) => provider["id"] === idp.identityProvider
|
||
|
)?.groupName!;
|
||
|
return (
|
||
|
<Label color={groupName === "Social" ? "blue" : "orange"}>
|
||
|
{groupName === "Social" ? "Social" : "Custom"}
|
||
|
</Label>
|
||
|
);
|
||
|
};
|
||
|
|
||
|
const badgeRenderer2 = (idp: IdentityProviderRepresentation) => {
|
||
|
const groupName = identityProviders?.find(
|
||
|
(provider) => provider["id"] === idp.providerId
|
||
|
)?.groupName!;
|
||
|
return (
|
||
|
<Label color={groupName === "User-defined" ? "orange" : "blue"}>
|
||
|
{groupName === "User-defined" ? "Custom" : groupName!}
|
||
|
</Label>
|
||
|
);
|
||
|
};
|
||
|
|
||
|
const unlinkRenderer = (fedIdentity: FederatedIdentityRepresentation) => {
|
||
|
return (
|
||
|
<Button
|
||
|
variant="link"
|
||
|
onClick={() => {
|
||
|
setFederatedId(fedIdentity.identityProvider!);
|
||
|
toggleUnlinkDialog();
|
||
|
}}
|
||
|
>
|
||
|
{t("unlinkAccount")}
|
||
|
</Button>
|
||
|
);
|
||
|
};
|
||
|
|
||
|
const linkRenderer = (idp: IdentityProviderRepresentation) => {
|
||
|
return (
|
||
|
<Button
|
||
|
variant="link"
|
||
|
onClick={() => {
|
||
|
setFederatedId(idp.alias!);
|
||
|
setIsLinkIdPModalOpen(true);
|
||
|
}}
|
||
|
>
|
||
|
{t("linkAccount")}
|
||
|
</Button>
|
||
|
);
|
||
|
};
|
||
|
|
||
|
return (
|
||
|
<>
|
||
|
{isLinkIdPModalOpen && (
|
||
|
<UserIdpModal
|
||
|
federatedId={federatedId}
|
||
|
handleModalToggle={handleModalToggle}
|
||
|
refresh={refresh}
|
||
|
/>
|
||
|
)}
|
||
|
<UnlinkConfirm />
|
||
|
<PageSection variant="light">
|
||
|
<FormPanel title={t("linkedIdPs")} className="kc-linked-idps">
|
||
|
<TextContent>
|
||
|
<Text className="kc-available-idps-text">
|
||
|
{t("linkedIdPsText")}
|
||
|
</Text>
|
||
|
</TextContent>
|
||
|
<KeycloakDataTable
|
||
|
loader={linkedIdPsLoader}
|
||
|
key={key}
|
||
|
isPaginated={false}
|
||
|
ariaLabelKey="users:LinkedIdPs"
|
||
|
className="kc-linked-IdPs-table"
|
||
|
columns={[
|
||
|
{
|
||
|
name: "identityProvider",
|
||
|
displayKey: "common:name",
|
||
|
cellFormatters: [emptyFormatter()],
|
||
|
cellRenderer: idpLinkRenderer,
|
||
|
transforms: [cellWidth(20)],
|
||
|
},
|
||
|
{
|
||
|
name: "type",
|
||
|
displayKey: "common:type",
|
||
|
cellFormatters: [emptyFormatter()],
|
||
|
cellRenderer: badgeRenderer1,
|
||
|
transforms: [cellWidth(10)],
|
||
|
},
|
||
|
{
|
||
|
name: "userId",
|
||
|
displayKey: "users:userID",
|
||
|
cellFormatters: [emptyFormatter()],
|
||
|
transforms: [cellWidth(30)],
|
||
|
},
|
||
|
{
|
||
|
name: "userName",
|
||
|
displayKey: "users:username",
|
||
|
cellFormatters: [emptyFormatter()],
|
||
|
transforms: [cellWidth(20)],
|
||
|
},
|
||
|
{
|
||
|
name: "",
|
||
|
cellFormatters: [emptyFormatter()],
|
||
|
cellRenderer: unlinkRenderer,
|
||
|
transforms: [cellWidth(20)],
|
||
|
},
|
||
|
]}
|
||
|
emptyState={
|
||
|
<TextContent className="kc-no-providers-text">
|
||
|
<Text>{t("users:noProvidersLinked")}</Text>
|
||
|
</TextContent>
|
||
|
}
|
||
|
/>
|
||
|
</FormPanel>
|
||
|
<FormPanel className="kc-available-idps" title={t("availableIdPs")}>
|
||
|
<TextContent>
|
||
|
<Text className="kc-available-idps-text">
|
||
|
{t("availableIdPsText")}
|
||
|
</Text>
|
||
|
</TextContent>
|
||
|
<KeycloakDataTable
|
||
|
loader={availableIdPsLoader}
|
||
|
key={key}
|
||
|
isPaginated={false}
|
||
|
ariaLabelKey="users:LinkedIdPs"
|
||
|
className="kc-linked-IdPs-table"
|
||
|
columns={[
|
||
|
{
|
||
|
name: "alias",
|
||
|
displayKey: "common:name",
|
||
|
cellFormatters: [emptyFormatter(), upperCaseFormatter()],
|
||
|
transforms: [cellWidth(20)],
|
||
|
},
|
||
|
{
|
||
|
name: "type",
|
||
|
displayKey: "common:type",
|
||
|
cellFormatters: [emptyFormatter()],
|
||
|
cellRenderer: badgeRenderer2,
|
||
|
transforms: [cellWidth(60)],
|
||
|
},
|
||
|
{
|
||
|
name: "",
|
||
|
cellFormatters: [emptyFormatter()],
|
||
|
cellRenderer: linkRenderer,
|
||
|
},
|
||
|
]}
|
||
|
emptyState={
|
||
|
<TextContent className="kc-no-providers-text">
|
||
|
<Text>{t("users:noAvailableIdentityProviders")}</Text>
|
||
|
</TextContent>
|
||
|
}
|
||
|
/>
|
||
|
</FormPanel>
|
||
|
</PageSection>
|
||
|
</>
|
||
|
);
|
||
|
};
|