2022-02-09 14:39:10 +00:00
|
|
|
import { useTranslation } from "react-i18next";
|
|
|
|
import { Controller, useForm } from "react-hook-form";
|
|
|
|
import {
|
|
|
|
AlertVariant,
|
|
|
|
ButtonVariant,
|
|
|
|
Form,
|
|
|
|
FormGroup,
|
|
|
|
Switch,
|
|
|
|
ValidatedOptions,
|
|
|
|
} from "@patternfly/react-core";
|
|
|
|
|
|
|
|
import type UserRepresentation from "@keycloak/keycloak-admin-client/lib/defs/userRepresentation";
|
|
|
|
import { HelpItem } from "../../components/help-enabler/HelpItem";
|
|
|
|
import { PasswordInput } from "../../components/password-input/PasswordInput";
|
|
|
|
import {
|
|
|
|
ConfirmDialogModal,
|
|
|
|
useConfirmDialog,
|
|
|
|
} from "../../components/confirm-dialog/ConfirmDialog";
|
|
|
|
import { useAdminClient } from "../../context/auth/AdminClient";
|
|
|
|
import { useAlerts } from "../../components/alert/Alerts";
|
|
|
|
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,
|
|
|
|
} = useForm<CredentialsForm>({
|
|
|
|
defaultValues: credFormDefaultValues,
|
|
|
|
mode: "onChange",
|
|
|
|
shouldUnregister: false,
|
|
|
|
});
|
|
|
|
|
|
|
|
const [confirm, toggle] = useToggle(true);
|
|
|
|
const password = watch("password", "");
|
|
|
|
|
2022-07-14 13:02:28 +00:00
|
|
|
const { adminClient } = useAdminClient();
|
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-03-28 13:32:06 +00:00
|
|
|
const { id } = (
|
|
|
|
await adminClient.users.getCredentials({ id: user.id! })
|
|
|
|
).find((c) => c.type === "password")!;
|
|
|
|
await adminClient.users.updateCredentialLabel(
|
|
|
|
{
|
|
|
|
id: user.id!,
|
|
|
|
credentialId: 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();
|
|
|
|
};
|
|
|
|
|
|
|
|
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"
|
|
|
|
name="password"
|
|
|
|
aria-label="password"
|
|
|
|
ref={register({ required: true })}
|
|
|
|
/>
|
|
|
|
</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"
|
|
|
|
name="passwordConfirmation"
|
|
|
|
aria-label="passwordConfirm"
|
|
|
|
ref={register({
|
|
|
|
required: true,
|
|
|
|
validate: (value) =>
|
|
|
|
value === password ||
|
|
|
|
t("confirmPasswordDoesNotMatch").toString(),
|
|
|
|
})}
|
|
|
|
/>
|
|
|
|
</FormGroup>
|
|
|
|
<FormGroup
|
|
|
|
label={t("common:temporaryPassword")}
|
|
|
|
labelIcon={
|
|
|
|
<HelpItem
|
|
|
|
helpText="temporaryPasswordHelpText"
|
|
|
|
fieldLabelId="temporaryPassword"
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
fieldId="kc-temporaryPassword"
|
|
|
|
>
|
|
|
|
<Controller
|
|
|
|
name="temporaryPassword"
|
|
|
|
defaultValue={true}
|
|
|
|
control={control}
|
|
|
|
render={({ onChange, value }) => (
|
|
|
|
<Switch
|
|
|
|
className="kc-temporaryPassword"
|
|
|
|
onChange={onChange}
|
|
|
|
isChecked={value}
|
|
|
|
label={t("common:on")}
|
|
|
|
labelOff={t("common:off")}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
</FormGroup>
|
|
|
|
</Form>
|
|
|
|
</ConfirmDialogModal>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
};
|