2023-01-26 09:31:07 +00:00
|
|
|
import type UserRepresentation from "@keycloak/keycloak-admin-client/lib/defs/userRepresentation";
|
2022-02-09 14:39:10 +00:00
|
|
|
import {
|
|
|
|
AlertVariant,
|
|
|
|
ButtonVariant,
|
|
|
|
Form,
|
|
|
|
FormGroup,
|
|
|
|
Switch,
|
|
|
|
ValidatedOptions,
|
|
|
|
} from "@patternfly/react-core";
|
2023-01-26 09:31:07 +00:00
|
|
|
import { Controller, useForm } from "react-hook-form";
|
|
|
|
import { useTranslation } from "react-i18next";
|
2023-05-03 13:51:02 +00:00
|
|
|
import { HelpItem } from "ui-shared";
|
2022-02-09 14:39:10 +00:00
|
|
|
|
2023-05-03 13:51:02 +00:00
|
|
|
import { adminClient } from "../../admin-client";
|
2023-01-26 09:31:07 +00:00
|
|
|
import { useAlerts } from "../../components/alert/Alerts";
|
2022-02-09 14:39:10 +00:00
|
|
|
import {
|
|
|
|
ConfirmDialogModal,
|
|
|
|
useConfirmDialog,
|
|
|
|
} from "../../components/confirm-dialog/ConfirmDialog";
|
2023-01-26 09:31:07 +00:00
|
|
|
import { PasswordInput } from "../../components/password-input/PasswordInput";
|
2022-02-09 14:39:10 +00:00
|
|
|
import useToggle from "../../utils/useToggle";
|
|
|
|
|
|
|
|
type ResetPasswordDialogProps = {
|
|
|
|
user: UserRepresentation;
|
|
|
|
isResetPassword: boolean;
|
|
|
|
refresh: () => void;
|
|
|
|
onClose: () => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
export type CredentialsForm = {
|
|
|
|
password: string;
|
|
|
|
passwordConfirmation: string;
|
|
|
|
temporaryPassword: boolean;
|
|
|
|
};
|
|
|
|
|
|
|
|
const credFormDefaultValues: CredentialsForm = {
|
|
|
|
password: "",
|
|
|
|
passwordConfirmation: "",
|
|
|
|
temporaryPassword: true,
|
|
|
|
};
|
|
|
|
|
|
|
|
export const ResetPasswordDialog = ({
|
|
|
|
user,
|
|
|
|
isResetPassword,
|
|
|
|
refresh,
|
|
|
|
onClose,
|
|
|
|
}: ResetPasswordDialogProps) => {
|
|
|
|
const { t } = useTranslation("users");
|
|
|
|
const {
|
|
|
|
register,
|
|
|
|
control,
|
2022-04-08 12:37:31 +00:00
|
|
|
formState: { isValid, errors },
|
2022-02-09 14:39:10 +00:00
|
|
|
watch,
|
|
|
|
handleSubmit,
|
2023-05-26 09:57:57 +00:00
|
|
|
clearErrors,
|
|
|
|
setError,
|
2022-02-09 14:39:10 +00:00
|
|
|
} = useForm<CredentialsForm>({
|
|
|
|
defaultValues: credFormDefaultValues,
|
|
|
|
mode: "onChange",
|
|
|
|
});
|
|
|
|
|
|
|
|
const [confirm, toggle] = useToggle(true);
|
|
|
|
const password = watch("password", "");
|
2023-05-26 09:57:57 +00:00
|
|
|
const passwordConfirmation = watch("passwordConfirmation", "");
|
2022-02-09 14:39:10 +00:00
|
|
|
|
|
|
|
const { addAlert, addError } = useAlerts();
|
|
|
|
|
|
|
|
const [toggleConfirmSaveModal, ConfirmSaveModal] = useConfirmDialog({
|
|
|
|
titleKey: isResetPassword
|
|
|
|
? "users:resetPasswordConfirm"
|
|
|
|
: "users:setPasswordConfirm",
|
|
|
|
messageKey: isResetPassword
|
|
|
|
? t("resetPasswordConfirmText", { username: user.username })
|
|
|
|
: t("setPasswordConfirmText", { username: user.username }),
|
|
|
|
continueButtonLabel: isResetPassword
|
|
|
|
? "users:resetPassword"
|
|
|
|
: "users:savePassword",
|
|
|
|
continueButtonVariant: ButtonVariant.danger,
|
|
|
|
onConfirm: () => handleSubmit(saveUserPassword)(),
|
|
|
|
});
|
|
|
|
|
|
|
|
const saveUserPassword = async ({
|
|
|
|
password,
|
|
|
|
temporaryPassword,
|
|
|
|
}: CredentialsForm) => {
|
|
|
|
try {
|
|
|
|
await adminClient.users.resetPassword({
|
|
|
|
id: user.id!,
|
|
|
|
credential: {
|
|
|
|
temporary: temporaryPassword,
|
|
|
|
type: "password",
|
|
|
|
value: password,
|
|
|
|
},
|
|
|
|
});
|
2022-10-10 19:47:07 +00:00
|
|
|
const credentials = await adminClient.users.getCredentials({
|
|
|
|
id: user.id!,
|
|
|
|
});
|
|
|
|
const credentialLabel = credentials.find((c) => c.type === "password");
|
|
|
|
if (credentialLabel) {
|
|
|
|
await adminClient.users.updateCredentialLabel(
|
|
|
|
{
|
|
|
|
id: user.id!,
|
|
|
|
credentialId: credentialLabel.id!,
|
|
|
|
},
|
|
|
|
t("defaultPasswordLabel")
|
|
|
|
);
|
|
|
|
}
|
2022-02-09 14:39:10 +00:00
|
|
|
addAlert(
|
|
|
|
isResetPassword
|
|
|
|
? t("resetCredentialsSuccess")
|
|
|
|
: t("savePasswordSuccess"),
|
|
|
|
AlertVariant.success
|
|
|
|
);
|
|
|
|
refresh();
|
|
|
|
} catch (error) {
|
|
|
|
addError(
|
|
|
|
isResetPassword
|
|
|
|
? "users:resetPasswordError"
|
|
|
|
: "users:savePasswordError",
|
|
|
|
error
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
onClose();
|
|
|
|
};
|
|
|
|
|
2023-05-26 09:57:57 +00:00
|
|
|
const { onChange, ...rest } = register("password", { required: true });
|
2022-02-09 14:39:10 +00:00
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<ConfirmSaveModal />
|
|
|
|
<ConfirmDialogModal
|
|
|
|
titleKey={
|
|
|
|
isResetPassword
|
|
|
|
? t("resetPasswordFor", { username: user.username })
|
|
|
|
: t("setPasswordFor", { username: user.username })
|
|
|
|
}
|
|
|
|
open={confirm}
|
|
|
|
onCancel={onClose}
|
|
|
|
toggleDialog={toggle}
|
|
|
|
onConfirm={toggleConfirmSaveModal}
|
|
|
|
confirmButtonDisabled={!isValid}
|
|
|
|
continueButtonLabel="common:save"
|
|
|
|
>
|
|
|
|
<Form
|
|
|
|
id="userCredentials-form"
|
|
|
|
isHorizontal
|
|
|
|
className="keycloak__user-credentials__reset-form"
|
|
|
|
>
|
|
|
|
<FormGroup
|
|
|
|
name="password"
|
|
|
|
label={t("password")}
|
|
|
|
fieldId="password"
|
|
|
|
helperTextInvalid={t("common:required")}
|
|
|
|
validated={
|
|
|
|
errors.password
|
|
|
|
? ValidatedOptions.error
|
|
|
|
: ValidatedOptions.default
|
|
|
|
}
|
|
|
|
isRequired
|
|
|
|
>
|
|
|
|
<PasswordInput
|
|
|
|
data-testid="passwordField"
|
2023-01-26 09:31:07 +00:00
|
|
|
id="password"
|
2023-05-26 09:57:57 +00:00
|
|
|
onChange={(e) => {
|
|
|
|
onChange(e);
|
|
|
|
if (passwordConfirmation !== e.currentTarget.value) {
|
|
|
|
setError("passwordConfirmation", {
|
|
|
|
message: t("confirmPasswordDoesNotMatch").toString(),
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
clearErrors("passwordConfirmation");
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
{...rest}
|
2022-02-09 14:39:10 +00:00
|
|
|
/>
|
|
|
|
</FormGroup>
|
|
|
|
<FormGroup
|
|
|
|
name="passwordConfirmation"
|
|
|
|
label={
|
|
|
|
isResetPassword
|
|
|
|
? t("resetPasswordConfirmation")
|
|
|
|
: t("passwordConfirmation")
|
|
|
|
}
|
|
|
|
fieldId="passwordConfirmation"
|
|
|
|
helperTextInvalid={errors.passwordConfirmation?.message}
|
|
|
|
validated={
|
|
|
|
errors.passwordConfirmation
|
|
|
|
? ValidatedOptions.error
|
|
|
|
: ValidatedOptions.default
|
|
|
|
}
|
|
|
|
isRequired
|
|
|
|
>
|
|
|
|
<PasswordInput
|
|
|
|
data-testid="passwordConfirmationField"
|
2023-01-26 09:31:07 +00:00
|
|
|
id="passwordConfirmation"
|
|
|
|
{...register("passwordConfirmation", {
|
2022-02-09 14:39:10 +00:00
|
|
|
required: true,
|
|
|
|
validate: (value) =>
|
|
|
|
value === password ||
|
|
|
|
t("confirmPasswordDoesNotMatch").toString(),
|
|
|
|
})}
|
|
|
|
/>
|
|
|
|
</FormGroup>
|
|
|
|
<FormGroup
|
|
|
|
label={t("common:temporaryPassword")}
|
|
|
|
labelIcon={
|
|
|
|
<HelpItem
|
2023-03-07 09:29:40 +00:00
|
|
|
helpText={t("temporaryPasswordHelpText")}
|
2022-02-09 14:39:10 +00:00
|
|
|
fieldLabelId="temporaryPassword"
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
fieldId="kc-temporaryPassword"
|
|
|
|
>
|
|
|
|
<Controller
|
|
|
|
name="temporaryPassword"
|
|
|
|
defaultValue={true}
|
|
|
|
control={control}
|
2023-01-26 09:31:07 +00:00
|
|
|
render={({ field }) => (
|
2022-02-09 14:39:10 +00:00
|
|
|
<Switch
|
|
|
|
className="kc-temporaryPassword"
|
2023-01-26 09:31:07 +00:00
|
|
|
onChange={field.onChange}
|
|
|
|
isChecked={field.value}
|
2022-02-09 14:39:10 +00:00
|
|
|
label={t("common:on")}
|
|
|
|
labelOff={t("common:off")}
|
2022-08-30 13:07:51 +00:00
|
|
|
aria-label={t("common:temporaryPassword")}
|
2022-02-09 14:39:10 +00:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
</FormGroup>
|
|
|
|
</Form>
|
|
|
|
</ConfirmDialogModal>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
};
|