mirror of
https://github.com/labring/FastGPT.git
synced 2025-10-21 11:30:06 +00:00
feat: add captcha (#2613)
* feat: captcha * add borderRadius * code perf
This commit is contained in:
@@ -2,12 +2,14 @@ export enum UserAuthTypeEnum {
|
|||||||
register = 'register',
|
register = 'register',
|
||||||
findPassword = 'findPassword',
|
findPassword = 'findPassword',
|
||||||
wxLogin = 'wxLogin',
|
wxLogin = 'wxLogin',
|
||||||
bindNotification = 'bindNotification'
|
bindNotification = 'bindNotification',
|
||||||
|
captcha = 'captcha'
|
||||||
}
|
}
|
||||||
|
|
||||||
export const userAuthTypeMap = {
|
export const userAuthTypeMap = {
|
||||||
[UserAuthTypeEnum.register]: 'register',
|
[UserAuthTypeEnum.register]: 'register',
|
||||||
[UserAuthTypeEnum.findPassword]: 'findPassword',
|
[UserAuthTypeEnum.findPassword]: 'findPassword',
|
||||||
[UserAuthTypeEnum.wxLogin]: 'wxLogin',
|
[UserAuthTypeEnum.wxLogin]: 'wxLogin',
|
||||||
[UserAuthTypeEnum.bindNotification]: 'bindNotification'
|
[UserAuthTypeEnum.bindNotification]: 'bindNotification',
|
||||||
|
[UserAuthTypeEnum.captcha]: 'captcha'
|
||||||
};
|
};
|
||||||
|
@@ -647,8 +647,7 @@
|
|||||||
"success": "Start syncing"
|
"success": "Start syncing"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"training": {
|
"training": {}
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"data": {
|
"data": {
|
||||||
"Auxiliary Data": "Auxiliary data",
|
"Auxiliary Data": "Auxiliary data",
|
||||||
@@ -1257,6 +1256,7 @@
|
|||||||
"auth": {
|
"auth": {
|
||||||
"Sending Code": "Sending"
|
"Sending Code": "Sending"
|
||||||
},
|
},
|
||||||
|
"captcha_placeholder": "Please enter the verification code",
|
||||||
"inform": {
|
"inform": {
|
||||||
"System message": "System message"
|
"System message": "System message"
|
||||||
},
|
},
|
||||||
|
@@ -647,8 +647,7 @@
|
|||||||
"success": "开始同步"
|
"success": "开始同步"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"training": {
|
"training": {}
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"data": {
|
"data": {
|
||||||
"Auxiliary Data": "辅助数据",
|
"Auxiliary Data": "辅助数据",
|
||||||
@@ -1248,6 +1247,7 @@
|
|||||||
},
|
},
|
||||||
"user": {
|
"user": {
|
||||||
"Avatar": "头像",
|
"Avatar": "头像",
|
||||||
|
"captcha_placeholder": "请输入验证码",
|
||||||
"Go laf env": "点击前往 {{env}} 获取 PAT 凭证。",
|
"Go laf env": "点击前往 {{env}} 获取 PAT 凭证。",
|
||||||
"Laf account course": "查看绑定 laf 账号教程。",
|
"Laf account course": "查看绑定 laf 账号教程。",
|
||||||
"Laf account intro": "绑定你的 laf 账号后,你将可以在工作流中使用 laf 模块,实现在线编写代码。",
|
"Laf account intro": "绑定你的 laf 账号后,你将可以在工作流中使用 laf 模块,实现在线编写代码。",
|
||||||
|
@@ -0,0 +1,64 @@
|
|||||||
|
import { getCaptchaPic } from '@/web/support/user/api';
|
||||||
|
import { useSendCode } from '@/web/support/user/hooks/useSendCode';
|
||||||
|
import { Box, Button, Input, Image, ModalBody, ModalFooter } from '@chakra-ui/react';
|
||||||
|
import { UserAuthTypeEnum } from '@fastgpt/global/support/user/auth/constants';
|
||||||
|
import MyModal from '@fastgpt/web/components/common/MyModal';
|
||||||
|
import { useRequest2 } from '@fastgpt/web/hooks/useRequest';
|
||||||
|
import { useTranslation } from 'next-i18next';
|
||||||
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
const SendCodeAuthModal = ({
|
||||||
|
username,
|
||||||
|
type,
|
||||||
|
onClose
|
||||||
|
}: {
|
||||||
|
username: string;
|
||||||
|
type: UserAuthTypeEnum;
|
||||||
|
onClose: () => void;
|
||||||
|
}) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [captchaInput, setCaptchaInput] = useState('');
|
||||||
|
const { codeSending, sendCode } = useSendCode();
|
||||||
|
const {
|
||||||
|
data,
|
||||||
|
loading,
|
||||||
|
runAsync: getCaptcha
|
||||||
|
} = useRequest2(() => getCaptchaPic(username), { manual: false });
|
||||||
|
return (
|
||||||
|
<MyModal isOpen={true} isLoading={loading}>
|
||||||
|
<ModalBody pt={8}>
|
||||||
|
<Image
|
||||||
|
borderRadius={'md'}
|
||||||
|
w={'100%'}
|
||||||
|
h={'200px'}
|
||||||
|
_hover={{ cursor: 'pointer' }}
|
||||||
|
mb={8}
|
||||||
|
onClick={getCaptcha}
|
||||||
|
src={data?.captchaImage}
|
||||||
|
alt="captcha"
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
placeholder={t('common:support.user.captcha_placeholder')}
|
||||||
|
value={captchaInput}
|
||||||
|
onChange={(e) => setCaptchaInput(e.target.value)}
|
||||||
|
/>
|
||||||
|
</ModalBody>
|
||||||
|
<ModalFooter gap={2}>
|
||||||
|
<Button isLoading={codeSending} variant={'whiteBase'} onClick={onClose}>
|
||||||
|
{t('common:common.Cancel')}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
isLoading={codeSending}
|
||||||
|
onClick={async () => {
|
||||||
|
await sendCode({ username, type, captcha: captchaInput });
|
||||||
|
onClose();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{t('common:common.Confirm')}
|
||||||
|
</Button>
|
||||||
|
</ModalFooter>
|
||||||
|
</MyModal>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default SendCodeAuthModal;
|
@@ -1,5 +1,14 @@
|
|||||||
import React, { useCallback } from 'react';
|
import React, { useCallback } from 'react';
|
||||||
import { ModalBody, Box, Flex, Input, ModalFooter, Button, HStack } from '@chakra-ui/react';
|
import {
|
||||||
|
ModalBody,
|
||||||
|
Box,
|
||||||
|
Flex,
|
||||||
|
Input,
|
||||||
|
ModalFooter,
|
||||||
|
Button,
|
||||||
|
HStack,
|
||||||
|
useDisclosure
|
||||||
|
} from '@chakra-ui/react';
|
||||||
import MyModal from '@fastgpt/web/components/common/MyModal';
|
import MyModal from '@fastgpt/web/components/common/MyModal';
|
||||||
import { useTranslation } from 'next-i18next';
|
import { useTranslation } from 'next-i18next';
|
||||||
import { useForm } from 'react-hook-form';
|
import { useForm } from 'react-hook-form';
|
||||||
@@ -9,6 +18,8 @@ import Icon from '@fastgpt/web/components/common/Icon';
|
|||||||
import { useSendCode } from '@/web/support/user/hooks/useSendCode';
|
import { useSendCode } from '@/web/support/user/hooks/useSendCode';
|
||||||
import { useUserStore } from '@/web/support/user/useUserStore';
|
import { useUserStore } from '@/web/support/user/useUserStore';
|
||||||
import { useSystemStore } from '@/web/common/system/useSystemStore';
|
import { useSystemStore } from '@/web/common/system/useSystemStore';
|
||||||
|
import SendCodeAuthModal from '@/components/support/user/safe/SendCodeAuthModal';
|
||||||
|
import { UserAuthTypeEnum } from '@fastgpt/global/support/user/auth/constants';
|
||||||
|
|
||||||
type FormType = {
|
type FormType = {
|
||||||
account: string;
|
account: string;
|
||||||
@@ -27,7 +38,11 @@ const UpdateNotificationModal = ({ onClose }: { onClose: () => void }) => {
|
|||||||
});
|
});
|
||||||
const account = watch('account');
|
const account = watch('account');
|
||||||
const verifyCode = watch('verifyCode');
|
const verifyCode = watch('verifyCode');
|
||||||
|
const {
|
||||||
|
isOpen: openCodeAuthModal,
|
||||||
|
onOpen: onOpenCodeAuthModal,
|
||||||
|
onClose: onCloseCodeAuthModal
|
||||||
|
} = useDisclosure();
|
||||||
const { runAsync: onSubmit, loading: isLoading } = useRequest2(
|
const { runAsync: onSubmit, loading: isLoading } = useRequest2(
|
||||||
(data: FormType) => {
|
(data: FormType) => {
|
||||||
return updateNotificationAccount(data);
|
return updateNotificationAccount(data);
|
||||||
@@ -42,16 +57,13 @@ const UpdateNotificationModal = ({ onClose }: { onClose: () => void }) => {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const { sendCodeText, sendCode, codeCountDown } = useSendCode();
|
const { sendCodeText, codeCountDown } = useSendCode();
|
||||||
|
|
||||||
const onclickSendCode = useCallback(async () => {
|
const onclickSendCode = useCallback(async () => {
|
||||||
const check = await trigger('account');
|
const check = await trigger('account');
|
||||||
if (!check) return;
|
if (!check) return;
|
||||||
sendCode({
|
onOpenCodeAuthModal();
|
||||||
username: getValues('account'),
|
}, [onOpenCodeAuthModal, trigger]);
|
||||||
type: 'bindNotification'
|
|
||||||
});
|
|
||||||
}, [getValues, sendCode, trigger]);
|
|
||||||
|
|
||||||
const placeholder = feConfigs?.bind_notification_method
|
const placeholder = feConfigs?.bind_notification_method
|
||||||
?.map((item) => {
|
?.map((item) => {
|
||||||
@@ -65,6 +77,7 @@ const UpdateNotificationModal = ({ onClose }: { onClose: () => void }) => {
|
|||||||
.join('/');
|
.join('/');
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
<MyModal
|
<MyModal
|
||||||
isOpen
|
isOpen
|
||||||
iconSrc="common/settingLight"
|
iconSrc="common/settingLight"
|
||||||
@@ -127,6 +140,14 @@ const UpdateNotificationModal = ({ onClose }: { onClose: () => void }) => {
|
|||||||
</Button>
|
</Button>
|
||||||
</ModalFooter>
|
</ModalFooter>
|
||||||
</MyModal>
|
</MyModal>
|
||||||
|
{openCodeAuthModal && (
|
||||||
|
<SendCodeAuthModal
|
||||||
|
onClose={onCloseCodeAuthModal}
|
||||||
|
username={getValues('account')}
|
||||||
|
type={UserAuthTypeEnum.bindNotification}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import React, { useState, Dispatch, useCallback } from 'react';
|
import React, { useState, Dispatch, useCallback } from 'react';
|
||||||
import { FormControl, Box, Input, Button } from '@chakra-ui/react';
|
import { FormControl, Box, Input, Button, useDisclosure } from '@chakra-ui/react';
|
||||||
import { useForm } from 'react-hook-form';
|
import { useForm } from 'react-hook-form';
|
||||||
import { LoginPageTypeEnum } from '@/web/support/user/login/constants';
|
import { LoginPageTypeEnum } from '@/web/support/user/login/constants';
|
||||||
import { postFindPassword } from '@/web/support/user/api';
|
import { postFindPassword } from '@/web/support/user/api';
|
||||||
@@ -8,6 +8,8 @@ import type { ResLogin } from '@/global/support/api/userRes.d';
|
|||||||
import { useToast } from '@fastgpt/web/hooks/useToast';
|
import { useToast } from '@fastgpt/web/hooks/useToast';
|
||||||
import { useSystemStore } from '@/web/common/system/useSystemStore';
|
import { useSystemStore } from '@/web/common/system/useSystemStore';
|
||||||
import { useTranslation } from 'next-i18next';
|
import { useTranslation } from 'next-i18next';
|
||||||
|
import { UserAuthTypeEnum } from '@fastgpt/global/support/user/auth/constants';
|
||||||
|
import SendCodeAuthModal from '@/components/support/user/safe/SendCodeAuthModal';
|
||||||
interface Props {
|
interface Props {
|
||||||
setPageType: Dispatch<`${LoginPageTypeEnum}`>;
|
setPageType: Dispatch<`${LoginPageTypeEnum}`>;
|
||||||
loginSuccess: (e: ResLogin) => void;
|
loginSuccess: (e: ResLogin) => void;
|
||||||
@@ -34,16 +36,17 @@ const RegisterForm = ({ setPageType, loginSuccess }: Props) => {
|
|||||||
mode: 'onBlur'
|
mode: 'onBlur'
|
||||||
});
|
});
|
||||||
|
|
||||||
const { codeSending, sendCodeText, sendCode, codeCountDown } = useSendCode();
|
const { sendCodeText, codeCountDown } = useSendCode();
|
||||||
|
const {
|
||||||
|
isOpen: openCodeAuthModal,
|
||||||
|
onOpen: onOpenCodeAuthModal,
|
||||||
|
onClose: onCloseCodeAuthModal
|
||||||
|
} = useDisclosure();
|
||||||
const onclickSendCode = useCallback(async () => {
|
const onclickSendCode = useCallback(async () => {
|
||||||
const check = await trigger('username');
|
const check = await trigger('username');
|
||||||
if (!check) return;
|
if (!check) return;
|
||||||
sendCode({
|
onOpenCodeAuthModal();
|
||||||
username: getValues('username'),
|
}, [onOpenCodeAuthModal, trigger]);
|
||||||
type: 'findPassword'
|
|
||||||
});
|
|
||||||
}, [getValues, sendCode, trigger]);
|
|
||||||
|
|
||||||
const [requesting, setRequesting] = useState(false);
|
const [requesting, setRequesting] = useState(false);
|
||||||
const placeholder = feConfigs?.find_password_method
|
const placeholder = feConfigs?.find_password_method
|
||||||
@@ -200,6 +203,13 @@ const RegisterForm = ({ setPageType, loginSuccess }: Props) => {
|
|||||||
{t('user:password.to_login')}
|
{t('user:password.to_login')}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
|
{openCodeAuthModal && (
|
||||||
|
<SendCodeAuthModal
|
||||||
|
onClose={onCloseCodeAuthModal}
|
||||||
|
username={getValues('username')}
|
||||||
|
type={UserAuthTypeEnum.findPassword}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import React, { useState, Dispatch, useCallback } from 'react';
|
import React, { useState, Dispatch, useCallback } from 'react';
|
||||||
import { FormControl, Box, Input, Button } from '@chakra-ui/react';
|
import { FormControl, Box, Input, Button, useDisclosure } from '@chakra-ui/react';
|
||||||
import { useForm } from 'react-hook-form';
|
import { useForm } from 'react-hook-form';
|
||||||
import { LoginPageTypeEnum } from '@/web/support/user/login/constants';
|
import { LoginPageTypeEnum } from '@/web/support/user/login/constants';
|
||||||
import { postRegister } from '@/web/support/user/api';
|
import { postRegister } from '@/web/support/user/api';
|
||||||
@@ -11,6 +11,9 @@ import { emptyTemplates } from '@/web/core/app/templates';
|
|||||||
import { useSystemStore } from '@/web/common/system/useSystemStore';
|
import { useSystemStore } from '@/web/common/system/useSystemStore';
|
||||||
import { useTranslation } from 'next-i18next';
|
import { useTranslation } from 'next-i18next';
|
||||||
import { AppTypeEnum } from '@fastgpt/global/core/app/constants';
|
import { AppTypeEnum } from '@fastgpt/global/core/app/constants';
|
||||||
|
import { useRequest2 } from '@fastgpt/web/hooks/useRequest';
|
||||||
|
import SendCodeAuthModal from '@/components/support/user/safe/SendCodeAuthModal';
|
||||||
|
import { UserAuthTypeEnum } from '@fastgpt/global/support/user/auth/constants';
|
||||||
interface Props {
|
interface Props {
|
||||||
loginSuccess: (e: ResLogin) => void;
|
loginSuccess: (e: ResLogin) => void;
|
||||||
setPageType: Dispatch<`${LoginPageTypeEnum}`>;
|
setPageType: Dispatch<`${LoginPageTypeEnum}`>;
|
||||||
@@ -37,17 +40,18 @@ const RegisterForm = ({ setPageType, loginSuccess }: Props) => {
|
|||||||
} = useForm<RegisterType>({
|
} = useForm<RegisterType>({
|
||||||
mode: 'onBlur'
|
mode: 'onBlur'
|
||||||
});
|
});
|
||||||
|
const {
|
||||||
const { sendCodeText, sendCode, codeCountDown } = useSendCode();
|
isOpen: openCodeAuthModal,
|
||||||
|
onOpen: onOpenCodeAuthModal,
|
||||||
|
onClose: onCloseCodeAuthModal
|
||||||
|
} = useDisclosure();
|
||||||
|
const { sendCodeText, codeCountDown } = useSendCode();
|
||||||
|
|
||||||
const onclickSendCode = useCallback(async () => {
|
const onclickSendCode = useCallback(async () => {
|
||||||
const check = await trigger('username');
|
const check = await trigger('username');
|
||||||
if (!check) return;
|
if (!check) return;
|
||||||
sendCode({
|
onOpenCodeAuthModal();
|
||||||
username: getValues('username'),
|
}, [onOpenCodeAuthModal, trigger]);
|
||||||
type: 'register'
|
|
||||||
});
|
|
||||||
}, [getValues, sendCode, trigger]);
|
|
||||||
|
|
||||||
const [requesting, setRequesting] = useState(false);
|
const [requesting, setRequesting] = useState(false);
|
||||||
|
|
||||||
@@ -215,6 +219,13 @@ const RegisterForm = ({ setPageType, loginSuccess }: Props) => {
|
|||||||
{t('user:register.to_login')}
|
{t('user:register.to_login')}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
|
{openCodeAuthModal && (
|
||||||
|
<SendCodeAuthModal
|
||||||
|
onClose={onCloseCodeAuthModal}
|
||||||
|
username={getValues('username')}
|
||||||
|
type={UserAuthTypeEnum.register}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@@ -15,6 +15,7 @@ export const sendAuthCode = (data: {
|
|||||||
username: string;
|
username: string;
|
||||||
type: `${UserAuthTypeEnum}`;
|
type: `${UserAuthTypeEnum}`;
|
||||||
googleToken: string;
|
googleToken: string;
|
||||||
|
captcha: string;
|
||||||
}) => POST(`/proApi/support/user/inform/sendAuthCode`, data);
|
}) => POST(`/proApi/support/user/inform/sendAuthCode`, data);
|
||||||
|
|
||||||
export const getTokenLogin = () =>
|
export const getTokenLogin = () =>
|
||||||
@@ -82,3 +83,8 @@ export const getWXLoginQR = () =>
|
|||||||
|
|
||||||
export const getWXLoginResult = (code: string) =>
|
export const getWXLoginResult = (code: string) =>
|
||||||
GET<ResLogin>(`/proApi/support/user/account/login/wx/getResult`, { code });
|
GET<ResLogin>(`/proApi/support/user/account/login/wx/getResult`, { code });
|
||||||
|
|
||||||
|
export const getCaptchaPic = (username: string) =>
|
||||||
|
GET<{
|
||||||
|
captchaImage: string;
|
||||||
|
}>('/proApi/support/user/account/captcha', { username });
|
||||||
|
@@ -12,10 +12,18 @@ export const useSendCode = () => {
|
|||||||
const [codeCountDown, setCodeCountDown] = useState(0);
|
const [codeCountDown, setCodeCountDown] = useState(0);
|
||||||
|
|
||||||
const { runAsync: sendCode, loading: codeSending } = useRequest2(
|
const { runAsync: sendCode, loading: codeSending } = useRequest2(
|
||||||
async ({ username, type }: { username: string; type: `${UserAuthTypeEnum}` }) => {
|
async ({
|
||||||
|
username,
|
||||||
|
type,
|
||||||
|
captcha
|
||||||
|
}: {
|
||||||
|
username: string;
|
||||||
|
type: `${UserAuthTypeEnum}`;
|
||||||
|
captcha: string;
|
||||||
|
}) => {
|
||||||
if (codeCountDown > 0) return;
|
if (codeCountDown > 0) return;
|
||||||
const googleToken = await getClientToken(feConfigs.googleClientVerKey);
|
const googleToken = await getClientToken(feConfigs.googleClientVerKey);
|
||||||
await sendAuthCode({ username, type, googleToken });
|
await sendAuthCode({ username, type, googleToken, captcha });
|
||||||
setCodeCountDown(60);
|
setCodeCountDown(60);
|
||||||
|
|
||||||
timer = setInterval(() => {
|
timer = setInterval(() => {
|
||||||
|
Reference in New Issue
Block a user