2022-08-30 19:46:52 +00:00
|
|
|
import { IdentityProvider } from "@prisma/client";
|
2022-12-07 20:53:44 +00:00
|
|
|
import { GetServerSidePropsContext } from "next";
|
2022-08-30 19:46:52 +00:00
|
|
|
import { Trans } from "next-i18next";
|
2023-01-31 20:44:14 +00:00
|
|
|
import { useState } from "react";
|
2022-09-15 16:59:48 +00:00
|
|
|
import { useForm } from "react-hook-form";
|
2022-08-30 19:46:52 +00:00
|
|
|
|
2023-01-05 17:00:16 +00:00
|
|
|
import { getLayout } from "@calcom/features/settings/layouts/SettingsLayout";
|
2022-09-06 18:23:17 +00:00
|
|
|
import { identityProviderNameMap } from "@calcom/lib/auth";
|
2022-08-30 19:46:52 +00:00
|
|
|
import { useLocale } from "@calcom/lib/hooks/useLocale";
|
2023-01-31 20:44:14 +00:00
|
|
|
import { userMetadata } from "@calcom/prisma/zod-utils";
|
2022-08-30 19:46:52 +00:00
|
|
|
import { trpc } from "@calcom/trpc/react";
|
2023-01-31 20:44:14 +00:00
|
|
|
import { Button, Form, Meta, PasswordField, Select, SettingsToggle, showToast } from "@calcom/ui";
|
2022-08-30 19:46:52 +00:00
|
|
|
|
2022-12-07 20:53:44 +00:00
|
|
|
import { ssrInit } from "@server/lib/ssr";
|
|
|
|
|
2023-01-31 20:44:14 +00:00
|
|
|
type ChangePasswordSessionFormValues = {
|
2022-09-15 16:59:48 +00:00
|
|
|
oldPassword: string;
|
|
|
|
newPassword: string;
|
2023-01-31 20:44:14 +00:00
|
|
|
sessionTimeout?: number;
|
2022-09-15 16:59:48 +00:00
|
|
|
};
|
|
|
|
|
2022-08-30 19:46:52 +00:00
|
|
|
const PasswordView = () => {
|
|
|
|
const { t } = useLocale();
|
2023-01-31 20:44:14 +00:00
|
|
|
const utils = trpc.useContext();
|
2022-11-10 23:40:01 +00:00
|
|
|
const { data: user } = trpc.viewer.me.useQuery();
|
2023-01-31 20:44:14 +00:00
|
|
|
const metadata = userMetadata.parse(user?.metadata);
|
2022-08-30 19:46:52 +00:00
|
|
|
|
2023-01-31 20:44:14 +00:00
|
|
|
const sessionMutation = trpc.viewer.updateProfile.useMutation({
|
|
|
|
onSuccess: () => {
|
|
|
|
showToast(t("session_timeout_changed"), "success");
|
|
|
|
formMethods.reset(formMethods.getValues());
|
|
|
|
},
|
|
|
|
onSettled: () => {
|
|
|
|
utils.viewer.me.invalidate();
|
|
|
|
},
|
|
|
|
onMutate: async ({ metadata }) => {
|
|
|
|
await utils.viewer.me.cancel();
|
|
|
|
const previousValue = utils.viewer.me.getData();
|
|
|
|
const previousMetadata = userMetadata.parse(previousValue?.metadata);
|
|
|
|
|
|
|
|
if (previousValue && metadata?.sessionTimeout) {
|
|
|
|
utils.viewer.me.setData(undefined, {
|
|
|
|
...previousValue,
|
|
|
|
metadata: { ...previousMetadata, sessionTimeout: metadata?.sessionTimeout },
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return { previousValue };
|
|
|
|
},
|
|
|
|
onError: (error, _, context) => {
|
|
|
|
if (context?.previousValue) {
|
|
|
|
utils.viewer.me.setData(undefined, context.previousValue);
|
|
|
|
}
|
|
|
|
showToast(`${t("session_timeout_change_error")}, ${error.message}`, "error");
|
|
|
|
},
|
|
|
|
});
|
|
|
|
const passwordMutation = trpc.viewer.auth.changePassword.useMutation({
|
2022-08-30 19:46:52 +00:00
|
|
|
onSuccess: () => {
|
2022-09-06 18:23:17 +00:00
|
|
|
showToast(t("password_has_been_changed"), "success");
|
2023-01-31 20:44:14 +00:00
|
|
|
formMethods.resetField("oldPassword");
|
|
|
|
formMethods.resetField("newPassword");
|
2022-08-30 19:46:52 +00:00
|
|
|
},
|
|
|
|
onError: (error) => {
|
|
|
|
showToast(`${t("error_updating_password")}, ${error.message}`, "error");
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2023-01-31 20:44:14 +00:00
|
|
|
const formMethods = useForm<ChangePasswordSessionFormValues>({
|
2022-09-15 16:59:48 +00:00
|
|
|
defaultValues: {
|
|
|
|
oldPassword: "",
|
|
|
|
newPassword: "",
|
2023-01-31 20:44:14 +00:00
|
|
|
sessionTimeout: metadata?.sessionTimeout,
|
2022-09-15 16:59:48 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2023-01-31 20:44:14 +00:00
|
|
|
const sessionTimeoutWatch = formMethods.watch("sessionTimeout");
|
2022-09-15 16:59:48 +00:00
|
|
|
|
2023-01-31 20:44:14 +00:00
|
|
|
const handleSubmit = (values: ChangePasswordSessionFormValues) => {
|
|
|
|
const { oldPassword, newPassword, sessionTimeout } = values;
|
|
|
|
if (oldPassword && newPassword) {
|
|
|
|
passwordMutation.mutate({ oldPassword, newPassword });
|
|
|
|
}
|
|
|
|
if (metadata?.sessionTimeout !== sessionTimeout) {
|
|
|
|
sessionMutation.mutate({ metadata: { ...metadata, sessionTimeout } });
|
|
|
|
}
|
2022-09-15 16:59:48 +00:00
|
|
|
};
|
2022-08-30 19:46:52 +00:00
|
|
|
|
2023-01-31 20:44:14 +00:00
|
|
|
const timeoutOptions = [5, 10, 15].map((mins) => ({
|
|
|
|
label: t("multiple_duration_mins", { count: mins }),
|
|
|
|
value: mins,
|
|
|
|
}));
|
|
|
|
|
|
|
|
const isDisabled = formMethods.formState.isSubmitting || !formMethods.formState.isDirty;
|
|
|
|
|
2022-08-30 19:46:52 +00:00
|
|
|
return (
|
|
|
|
<>
|
2022-09-29 16:19:03 +00:00
|
|
|
<Meta title={t("password")} description={t("password_description")} />
|
2022-08-30 19:46:52 +00:00
|
|
|
{user && user.identityProvider !== IdentityProvider.CAL ? (
|
|
|
|
<div>
|
|
|
|
<div className="mt-6">
|
|
|
|
<h2 className="font-cal text-lg font-medium leading-6 text-gray-900">
|
|
|
|
{t("account_managed_by_identity_provider", {
|
|
|
|
provider: identityProviderNameMap[user.identityProvider],
|
|
|
|
})}
|
|
|
|
</h2>
|
|
|
|
</div>
|
|
|
|
<p className="mt-1 text-sm text-gray-500">
|
|
|
|
{t("account_managed_by_identity_provider_description", {
|
|
|
|
provider: identityProviderNameMap[user.identityProvider],
|
|
|
|
})}
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
) : (
|
2022-09-29 16:19:03 +00:00
|
|
|
<Form form={formMethods} handleSubmit={handleSubmit}>
|
2022-09-05 21:08:59 +00:00
|
|
|
<div className="max-w-[38rem] sm:flex sm:space-x-4">
|
|
|
|
<div className="flex-grow">
|
2023-01-31 20:44:14 +00:00
|
|
|
<PasswordField {...formMethods.register("oldPassword")} label={t("old_password")} />
|
2022-09-05 21:08:59 +00:00
|
|
|
</div>
|
|
|
|
<div className="flex-grow">
|
2023-01-31 20:44:14 +00:00
|
|
|
<PasswordField {...formMethods.register("newPassword")} label={t("new_password")} />
|
2022-09-05 21:08:59 +00:00
|
|
|
</div>
|
2022-08-30 19:46:52 +00:00
|
|
|
</div>
|
2023-01-31 20:44:14 +00:00
|
|
|
<p className="mt-4 max-w-[38rem] text-sm text-gray-600">
|
2022-09-29 16:19:03 +00:00
|
|
|
<Trans i18nKey="invalid_password_hint">
|
2022-08-30 19:46:52 +00:00
|
|
|
Password must be at least at least 7 characters, mix of uppercase & lowercase letters, and
|
2023-01-31 20:44:14 +00:00
|
|
|
contain at least 1 number.
|
2022-08-30 19:46:52 +00:00
|
|
|
</Trans>
|
|
|
|
</p>
|
2023-01-31 20:44:14 +00:00
|
|
|
<div className="mt-8 border-t border-gray-200 py-8">
|
|
|
|
<SettingsToggle
|
|
|
|
title={t("session_timeout")}
|
|
|
|
description={t("session_timeout_description")}
|
|
|
|
checked={sessionTimeoutWatch !== undefined}
|
|
|
|
data-testid="session-check"
|
|
|
|
onCheckedChange={(e) => {
|
|
|
|
if (!e) {
|
|
|
|
formMethods.setValue("sessionTimeout", undefined, { shouldDirty: true });
|
|
|
|
} else {
|
|
|
|
formMethods.setValue("sessionTimeout", 10, { shouldDirty: true });
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
{sessionTimeoutWatch && (
|
|
|
|
<div className="mt-4 text-sm">
|
|
|
|
<div className="flex items-center">
|
|
|
|
<p className="text-neutral-900 ltr:mr-2 rtl:ml-2">{t("session_timeout_after")}</p>
|
|
|
|
<Select
|
|
|
|
options={timeoutOptions}
|
|
|
|
defaultValue={
|
|
|
|
metadata?.sessionTimeout
|
|
|
|
? timeoutOptions.find((tmo) => tmo.value === metadata.sessionTimeout)
|
|
|
|
: timeoutOptions[1]
|
|
|
|
}
|
|
|
|
isSearchable={false}
|
|
|
|
className="block h-[36px] !w-auto min-w-0 flex-none rounded-md text-sm"
|
|
|
|
onChange={(event) => {
|
|
|
|
formMethods.setValue("sessionTimeout", event?.value, { shouldDirty: true });
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</div>
|
2022-09-15 16:59:48 +00:00
|
|
|
{/* TODO: Why is this Form not submitting? Hacky fix but works */}
|
|
|
|
<Button
|
|
|
|
color="primary"
|
|
|
|
className="mt-8"
|
2022-09-29 16:19:03 +00:00
|
|
|
type="submit"
|
2023-01-31 20:44:14 +00:00
|
|
|
disabled={isDisabled || passwordMutation.isLoading || sessionMutation.isLoading}>
|
2022-08-30 19:46:52 +00:00
|
|
|
{t("update")}
|
|
|
|
</Button>
|
|
|
|
</Form>
|
|
|
|
)}
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
PasswordView.getLayout = getLayout;
|
|
|
|
|
2022-12-07 20:53:44 +00:00
|
|
|
export const getServerSideProps = async (context: GetServerSidePropsContext) => {
|
|
|
|
const ssr = await ssrInit(context);
|
|
|
|
|
|
|
|
return {
|
|
|
|
props: {
|
|
|
|
trpcState: ssr.dehydrate(),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2022-08-30 19:46:52 +00:00
|
|
|
export default PasswordView;
|