cal.pub0.org/packages/features/ee/sso/components/ConnectionInfo.tsx

169 lines
5.0 KiB
TypeScript

import type { SSOConnection } from "@calcom/ee/sso/lib/saml";
import { APP_NAME } from "@calcom/lib/constants";
import { useLocale } from "@calcom/lib/hooks/useLocale";
import { trpc } from "@calcom/trpc/react";
import {
Button,
showToast,
Tooltip,
ConfirmationDialogContent,
Dialog,
DialogTrigger,
Label,
} from "@calcom/ui";
import { Clipboard } from "@calcom/ui/components/icon";
export default function ConnectionInfo({
teamId,
connection,
}: {
teamId: number | null;
connection: SSOConnection;
}) {
const { t } = useLocale();
const utils = trpc.useContext();
const connectionType = connection.type.toUpperCase();
// Delete SSO connection
const mutation = trpc.viewer.saml.delete.useMutation({
async onSuccess() {
showToast(
t("sso_connection_deleted_successfully", {
connectionType,
}),
"success"
);
await utils.viewer.saml.get.invalidate();
},
});
const deleteConnection = async () => {
mutation.mutate({
teamId,
});
};
return (
<div>
{connection.type === "saml" ? (
<SAMLInfo acsUrl={connection.acsUrl} entityId={connection.entityId} />
) : (
<OIDCInfo callbackUrl={connection.callbackUrl} />
)}
<hr className="border-subtle my-6" />
<div className="flex flex-col space-y-3">
<Label>{t("danger_zone")}</Label>
<Dialog>
<div>
<DialogTrigger asChild>
<Button color="destructive">{t("delete_sso_configuration", { connectionType })}</Button>
</DialogTrigger>
</div>
<ConfirmationDialogContent
variety="danger"
title={t("delete_sso_configuration", { connectionType })}
confirmBtnText={t("delete_sso_configuration_confirmation", { connectionType })}
onConfirm={deleteConnection}>
{t("delete_sso_configuration_confirmation_description", { appName: APP_NAME, connectionType })}
</ConfirmationDialogContent>
</Dialog>
</div>
</div>
);
}
// Connection info for SAML
const SAMLInfo = ({ acsUrl, entityId }: { acsUrl: string | null; entityId: string | null }) => {
const { t } = useLocale();
if (!acsUrl || !entityId) {
return null;
}
return (
<div className="space-y-6">
<div className="flex flex-col">
<div className="flex">
<Label>ACS URL</Label>
</div>
<div className="flex">
<code className="bg-subtle text-default flex w-full items-center truncate rounded rounded-r-none pl-2 font-mono">
{acsUrl}
</code>
<Tooltip side="top" content={t("copy_to_clipboard")}>
<Button
onClick={() => {
navigator.clipboard.writeText(acsUrl);
showToast(t("sso_saml_acsurl_copied"), "success");
}}
type="button"
className="rounded-l-none py-[19px] text-base ">
<Clipboard className="text-muted h-5 w-5 ltr:mr-2 rtl:ml-2" />
{t("copy")}
</Button>
</Tooltip>
</div>
</div>
<div className="flex flex-col">
<div className="flex">
<Label>Entity ID</Label>
</div>
<div className="flex">
<code className="bg-subtle text-default flex w-full items-center truncate rounded rounded-r-none pl-2 font-mono">
{entityId}
</code>
<Tooltip side="top" content={t("copy_to_clipboard")}>
<Button
onClick={() => {
navigator.clipboard.writeText(entityId);
showToast(t("sso_saml_entityid_copied"), "success");
}}
type="button"
className="rounded-l-none py-[19px] text-base ">
<Clipboard className="text-muted h-5 w-5 ltr:mr-2 rtl:ml-2" />
{t("copy")}
</Button>
</Tooltip>
</div>
</div>
</div>
);
};
// Connection info for OIDC
const OIDCInfo = ({ callbackUrl }: { callbackUrl: string | null }) => {
const { t } = useLocale();
if (!callbackUrl) {
return null;
}
return (
<div>
<div className="flex flex-col">
<div className="flex">
<Label>Callback URL</Label>
</div>
<div className="flex">
<code className="bg-subtle text-default flex w-full items-center truncate rounded rounded-r-none pl-2 font-mono">
{callbackUrl}
</code>
<Tooltip side="top" content={t("copy_to_clipboard")}>
<Button
onClick={() => {
navigator.clipboard.writeText(callbackUrl);
showToast(t("sso_oidc_callback_copied"), "success");
}}
type="button"
className="rounded-l-none py-[19px] text-base ">
<Clipboard className="text-muted h-5 w-5 ltr:mr-2 rtl:ml-2" />
{t("copy")}
</Button>
</Tooltip>
</div>
</div>
</div>
);
};