添加两步验证

This commit is contained in:
hanhh 2021-08-30 23:37:26 +08:00
parent 3a998a37f0
commit 86c3e9a843
11 changed files with 592 additions and 154 deletions

View File

@ -80,10 +80,14 @@ export default (app: Router) => {
async (req: Request, res: Response, next: NextFunction) => {
const logger: Logger = Container.get('logger');
try {
fs.readFile(config.authConfigFile, 'utf8', (err, data) => {
if (err) console.log(err);
const authInfo = JSON.parse(data);
res.send({ code: 200, data: { username: authInfo.username } });
const authService = Container.get(AuthService);
const authInfo = await authService.getUserInfo();
res.send({
code: 200,
data: {
username: authInfo.username,
twoFactorActived: authInfo.twoFactorActived,
},
});
} catch (e) {
logger.error('🔥 error: %o', e);
@ -91,4 +95,76 @@ export default (app: Router) => {
}
},
);
route.get(
'/user/two-factor/init',
async (req: Request, res: Response, next: NextFunction) => {
const logger: Logger = Container.get('logger');
try {
const authService = Container.get(AuthService);
const data = await authService.initTwoFactor();
res.send({ code: 200, data });
} catch (e) {
logger.error('🔥 error: %o', e);
return next(e);
}
},
);
route.put(
'/user/two-factor/active',
celebrate({
body: Joi.object({
code: Joi.string().required(),
}),
}),
async (req: Request, res: Response, next: NextFunction) => {
const logger: Logger = Container.get('logger');
try {
const authService = Container.get(AuthService);
const data = await authService.activeTwoFactor(req.body.code);
res.send({ code: 200, data });
} catch (e) {
logger.error('🔥 error: %o', e);
return next(e);
}
},
);
route.get(
'/user/two-factor/deactive',
async (req: Request, res: Response, next: NextFunction) => {
const logger: Logger = Container.get('logger');
try {
const authService = Container.get(AuthService);
const data = await authService.deactiveTwoFactor();
res.send({ code: 200, data });
} catch (e) {
logger.error('🔥 error: %o', e);
return next(e);
}
},
);
route.put(
'/user/two-factor/login',
celebrate({
body: Joi.object({
code: Joi.string().required(),
username: Joi.string().required(),
password: Joi.string().required(),
}),
}),
async (req: Request, res: Response, next: NextFunction) => {
const logger: Logger = Container.get('logger');
try {
const authService = Container.get(AuthService);
const data = await authService.twoFactorLogin(req.body, req);
res.send(data);
} catch (e) {
logger.error('🔥 error: %o', e);
return next(e);
}
},
);
};

View File

@ -18,7 +18,12 @@ export default ({ app }: { app: Application }) => {
app.use(bodyParser.urlencoded({ limit: '50mb', extended: true }));
app.use(
jwt({ secret: config.secret as string, algorithms: ['HS384'] }).unless({
path: ['/api/login', '/api/crons/status', /^\/open\//],
path: [
'/api/login',
'/api/crons/status',
/^\/open\//,
'/api/user/two-factor/login',
],
}),
);
@ -52,7 +57,9 @@ export default ({ app }: { app: Application }) => {
if (
!headerToken &&
req.path &&
(req.path === '/api/login' || req.path === '/open/auth/token')
(req.path === '/api/login' ||
req.path === '/open/auth/token' ||
req.path === '/api/user/two-factor/login')
) {
return next();
}

View File

@ -5,6 +5,7 @@ import config from '../config';
import * as fs from 'fs';
import _ from 'lodash';
import jwt from 'jsonwebtoken';
import { authenticator } from '@otplib/preset-default';
@Service()
export default class AuthService {
@ -32,6 +33,8 @@ export default class AuthService {
lastlogon,
lastip,
lastaddr,
twoFactorActived,
twoFactorChecked,
} = JSON.parse(content);
if (
@ -42,7 +45,21 @@ export default class AuthService {
return this.initAuthInfo();
}
if (twoFactorActived && !twoFactorChecked) {
return {
code: 420,
message: '请输入两步验证token',
};
}
if (retries > 2 && Date.now() - lastlogon < Math.pow(3, retries) * 1000) {
fs.writeFileSync(
config.authConfigFile,
JSON.stringify({
...JSON.parse(content),
twoFactorChecked: false,
}),
);
return {
code: 410,
message: `失败次数过多,请${Math.round(
@ -57,8 +74,9 @@ export default class AuthService {
const { ip, address } = await getNetIp(req);
if (username === cUsername && password === cPassword) {
const data = createRandomString(50, 100);
const expiration = twoFactorActived ? 30 : 3;
let token = jwt.sign({ data }, config.secret as any, {
expiresIn: 60 * 60 * 24 * 3,
expiresIn: 60 * 60 * 24 * expiration,
algorithm: 'HS384',
});
fs.writeFileSync(
@ -70,6 +88,7 @@ export default class AuthService {
retries: 0,
lastip: ip,
lastaddr: address,
twoFactorChecked: false,
}),
);
return { code: 200, data: { token, lastip, lastaddr, lastlogon } };
@ -82,6 +101,7 @@ export default class AuthService {
lastlogon: timestamp,
lastip: ip,
lastaddr: address,
twoFactorChecked: false,
}),
);
return { code: 400, message: config.authError };
@ -105,4 +125,68 @@ export default class AuthService {
message: '已初始化密码请前往auth.json查看并重新登录',
};
}
public getUserInfo(): Promise<any> {
return new Promise((resolve) => {
fs.readFile(config.authConfigFile, 'utf8', (err, data) => {
if (err) console.log(err);
resolve(JSON.parse(data));
});
});
}
public initTwoFactor() {
const secret = authenticator.generateSecret();
const authInfo = this.getAuthInfo();
const otpauth = authenticator.keyuri(authInfo.username, 'qinglong', secret);
this.updateAuthInfo(authInfo, { twoFactorSecret: secret });
return { secret, url: otpauth };
}
public activeTwoFactor(code: string) {
const authInfo = this.getAuthInfo();
const isValid = authenticator.verify({
token: code,
secret: authInfo.twoFactorSecret,
});
if (isValid) {
this.updateAuthInfo(authInfo, { twoFactorActived: true });
}
return isValid;
}
public twoFactorLogin({ username, password, code }, req) {
const authInfo = this.getAuthInfo();
const isValid = authenticator.verify({
token: code,
secret: authInfo.twoFactorSecret,
});
if (isValid) {
this.updateAuthInfo(authInfo, { twoFactorChecked: true });
return this.login({ username, password }, req);
} else {
return { code: 430, message: '验证失败' };
}
}
public deactiveTwoFactor() {
const authInfo = this.getAuthInfo();
this.updateAuthInfo(authInfo, {
twoFactorActived: false,
twoFactorSecret: '',
});
return true;
}
private getAuthInfo() {
const content = fs.readFileSync(config.authConfigFile, 'utf8');
return JSON.parse(content || '{}');
}
private updateAuthInfo(authInfo: any, info: any) {
fs.writeFileSync(
config.authConfigFile,
JSON.stringify({ ...authInfo, ...info }),
);
}
}

View File

@ -24,6 +24,7 @@
]
},
"dependencies": {
"@otplib/preset-default": "^12.0.1",
"body-parser": "^1.19.0",
"celebrate": "^13.0.3",
"cors": "^2.8.5",

View File

@ -4,6 +4,7 @@ specifiers:
'@ant-design/icons': ^4.6.2
'@ant-design/pro-layout': ^6.5.0
'@monaco-editor/react': ^4.2.1
'@otplib/preset-default': ^12.0.1
'@types/cors': ^2.8.10
'@types/express': ^4.17.8
'@types/express-jwt': ^6.0.1
@ -60,6 +61,7 @@ specifiers:
yorkie: ^2.0.0
dependencies:
'@otplib/preset-default': 12.0.1
body-parser: 1.19.0
celebrate: 13.0.4
cors: 2.8.5
@ -908,6 +910,31 @@ packages:
rimraf: 3.0.2
dev: true
/@otplib/core/12.0.1:
resolution: {integrity: sha512-4sGntwbA/AC+SbPhbsziRiD+jNDdIzsZ3JUyfZwjtKyc/wufl1pnSIaG4Uqx8ymPagujub0o92kgBnB89cuAMA==}
dev: false
/@otplib/plugin-crypto/12.0.1:
resolution: {integrity: sha512-qPuhN3QrT7ZZLcLCyKOSNhuijUi9G5guMRVrxq63r9YNOxxQjPm59gVxLM+7xGnHnM6cimY57tuKsjK7y9LM1g==}
dependencies:
'@otplib/core': 12.0.1
dev: false
/@otplib/plugin-thirty-two/12.0.1:
resolution: {integrity: sha512-MtT+uqRso909UkbrrYpJ6XFjj9D+x2Py7KjTO9JDPhL0bJUYVu5kFP4TFZW4NFAywrAtFRxOVY261u0qwb93gA==}
dependencies:
'@otplib/core': 12.0.1
thirty-two: 1.0.2
dev: false
/@otplib/preset-default/12.0.1:
resolution: {integrity: sha512-xf1v9oOJRyXfluBhMdpOkr+bsE+Irt+0D5uHtvg6x1eosfmHCsCC6ej/m7FXiWqdo0+ZUI6xSKDhJwc8yfiOPQ==}
dependencies:
'@otplib/core': 12.0.1
'@otplib/plugin-crypto': 12.0.1
'@otplib/plugin-thirty-two': 12.0.1
dev: false
/@qixian.cs/path-to-regexp/6.1.0:
resolution: {integrity: sha512-2jIiLiVZB1jnY7IIRQKtoV8Gnr7XIhk4mC88ONGunZE3hYt5IHUG4BE/6+JiTBjjEWQLBeWnZB8hGpppkufiVw==}
dev: true
@ -9084,6 +9111,11 @@ packages:
resolution: {integrity: sha512-uuVGNWzgJ4yhRaNSiubPY7OjISw4sw4E5Uv0wbjp+OzcbmVU/rsT8ujgcXJhn9ypzsgr5vlzpPqP+MBBKcGvbg==}
dev: false
/thirty-two/1.0.2:
resolution: {integrity: sha1-TKL//AKlEpDSdEueP1V2k8prYno=}
engines: {node: '>=0.2.6'}
dev: false
/throat/5.0.0:
resolution: {integrity: sha512-fcwX4mndzpLQKBS1DVYhGAcYaYt7vsHNIvQV+WXMvnow5cgjPphq5CaayLaGsjRdSCKZFNGt7/GYAuXaNOiYCA==}
dev: true

View File

@ -14,26 +14,6 @@ const titleMap: any = {
'/setting': '系统设置',
};
export function render(oldRender: any) {
request
.get(`${config.apiPrefix}user`)
.then((data) => {
if (data.data && data.data.username) {
return oldRender();
}
localStorage.removeItem(config.authKey);
history.push('/login');
oldRender();
})
.catch((e) => {
if (e.response && e.response.status === 401) {
localStorage.removeItem(config.authKey);
history.push('/login');
oldRender();
}
});
}
export function onRouteChange({ matchedRoutes }: any) {
if (matchedRoutes.length) {
const path: string = matchedRoutes[matchedRoutes.length - 1].route.path;

View File

@ -19,6 +19,7 @@ import { useCtx, useTheme } from '@/utils/hooks';
export default function (props: any) {
const ctx = useCtx();
const theme = useTheme();
const [user, setUser] = useState<any>();
const logout = () => {
request.post(`${config.apiPrefix}logout`).then(() => {
@ -27,12 +28,29 @@ export default function (props: any) {
});
};
const getUser = () => {
request
.get(`${config.apiPrefix}user`)
.then((data) => {
if (data.data.username) {
setUser(data.data);
}
})
.catch((e) => {
if (e.response && e.response.status === 401) {
localStorage.removeItem(config.authKey);
history.push('/login');
}
});
};
useEffect(() => {
const isAuth = localStorage.getItem(config.authKey);
if (!isAuth) {
history.push('/login');
}
vhCheck();
getUser();
// patch custome layout title as react node [object, object]
document.title = '控制面板';
@ -112,7 +130,7 @@ export default function (props: any) {
{...defaultProps}
>
{React.Children.map(props.children, (child) => {
return React.cloneElement(child, { ...ctx, ...theme });
return React.cloneElement(child, { ...ctx, ...theme, user });
})}
</ProLayout>
);

View File

@ -21,13 +21,11 @@
.content {
position: absolute;
top: 45%;
top: 86px;
left: 50%;
margin: -160px 0 0 -160px;
width: 320px;
height: 320px;
padding: 36px;
box-shadow: 0 0 100px rgba(0, 0, 0, 0.08);
margin-left: -170px;
width: 340px;
padding: 0 16px;
}
@media (min-width: @screen-md-min) {
@ -44,16 +42,20 @@
}
.header {
height: 44px;
line-height: 44px;
display: flex;
align-items: center;
justify-content: center;
flex-direction: column;
}
.logo {
width: 30px;
margin-right: 8px;
width: 48px;
display: block;
margin-bottom: 24px;
}
.title {
font-size: 20px;
margin-bottom: 16px;
}
.desc {
@ -64,7 +66,11 @@
}
.main {
margin: 35px auto 0;
padding: 20px;
border-radius: 6px;
background-color: #f6f8fa;
border: 1px solid #ebedef;
@media screen and (max-width: @screen-sm) {
width: 95%;
max-width: 320px;
@ -104,3 +110,7 @@
font-size: @font-size-base;
}
}
.extra {
margin-top: 20px;
}

View File

@ -13,6 +13,7 @@ import { history, Link } from 'umi';
import styles from './index.less';
import { request } from '@/utils/http';
import { useTheme } from '@/utils/hooks';
import { MobileOutlined } from '@ant-design/icons';
const FormItem = Form.Item;
const { Countdown } = Statistic;
@ -21,9 +22,13 @@ const Login = () => {
const [loading, setLoading] = useState(false);
const [waitTime, setWaitTime] = useState<any>();
const { theme } = useTheme();
const [twoFactor, setTwoFactor] = useState(false);
const [loginInfo, setLoginInfo] = useState<any>();
const [verifing, setVerifing] = useState(false);
const handleOk = (values: any) => {
setLoading(true);
setTwoFactor(false);
setWaitTime(null);
request
.post(`${config.apiPrefix}login`, {
@ -33,31 +38,14 @@ const Login = () => {
},
})
.then((data) => {
if (data.code === 200) {
const { token, lastip, lastaddr, lastlogon } = data.data;
localStorage.setItem(config.authKey, token);
notification.success({
message: '登录成功!',
description: (
<>
<div>
{lastlogon ? new Date(lastlogon).toLocaleString() : '-'}
</div>
<div>{lastaddr || '-'}</div>
<div>IP{lastip || '-'}</div>
</>
),
duration: 5,
if (data.code === 420) {
setLoginInfo({
username: values.username,
password: values.password,
});
history.push('/crontab');
} else if (data.code === 100) {
message.warn(data.message);
} else if (data.code === 410) {
message.error(data.message);
setWaitTime(data.data);
setTwoFactor(true);
} else {
message.error(data.message);
checkResponse(data);
}
setLoading(false);
})
@ -67,6 +55,55 @@ const Login = () => {
});
};
const completeTowFactor = (values: any) => {
setVerifing(true);
request
.put(`${config.apiPrefix}user/two-factor/login`, {
data: { ...loginInfo, code: values.code },
})
.then((data: any) => {
if (data.code === 430) {
message.error(data.message);
} else {
checkResponse(data);
}
setVerifing(false);
})
.catch((error: any) => {
console.log(error);
setVerifing(false);
});
};
const checkResponse = (data: any) => {
if (data.code === 200) {
const { token, lastip, lastaddr, lastlogon } = data.data;
localStorage.setItem(config.authKey, token);
notification.success({
message: '登录成功!',
description: (
<>
<div>
{lastlogon ? new Date(lastlogon).toLocaleString() : '-'}
</div>
<div>{lastaddr || '-'}</div>
<div>IP{lastip || '-'}</div>
</>
),
duration: 5,
});
history.push('/crontab');
} else if (data.code === 100) {
message.warn(data.message);
} else if (data.code === 410) {
message.error(data.message);
setWaitTime(data.data);
} else {
message.error(data.message);
}
};
useEffect(() => {
const isAuth = localStorage.getItem(config.authKey);
if (isAuth) {
@ -84,55 +121,88 @@ const Login = () => {
className={styles.logo}
src="/images/qinglong.png"
/>
<span className={styles.title}>{config.siteName}</span>
<span className={styles.title}>
{twoFactor ? '两步验证' : config.siteName}
</span>
</div>
</div>
<div className={styles.main}>
<Form onFinish={handleOk}>
<FormItem
name="username"
rules={[{ required: true, message: '请输入用户名' }]}
hasFeedback
>
<Input placeholder="用户名" autoFocus />
</FormItem>
<FormItem
name="password"
rules={[{ required: true, message: '请输入密码' }]}
hasFeedback
>
<Input type="password" placeholder="密码" />
</FormItem>
<Row>
{waitTime ? (
<Button type="primary" style={{ width: '100%' }} disabled>
<Countdown
valueStyle={{
color:
theme === 'vs'
? 'rgba(0,0,0,.25)'
: 'rgba(232, 230, 227, 0.25)',
}}
className="inline-countdown"
onFinish={() => setWaitTime(null)}
format="ss"
value={Date.now() + 1000 * waitTime}
/>
</Button>
) : (
<Button
type="primary"
htmlType="submit"
style={{ width: '100%' }}
loading={loading}
>
</Button>
)}
</Row>
</Form>
{twoFactor ? (
<Form layout="vertical" onFinish={completeTowFactor}>
<FormItem
name="code"
label="验证码"
rules={[
{
pattern: /^[0-9]{6}$/,
message: '验证码为6位数字',
validateTrigger: 'onBlur',
},
]}
hasFeedback
>
<Input placeholder="6位数字" autoComplete="off" />
</FormItem>
<Button
type="primary"
htmlType="submit"
style={{ width: '100%' }}
loading={verifing}
>
</Button>
</Form>
) : (
<Form layout="vertical" onFinish={handleOk}>
<FormItem name="username" label="用户名" hasFeedback>
<Input placeholder="用户名" autoFocus />
</FormItem>
<FormItem name="password" label="密码" hasFeedback>
<Input type="password" placeholder="密码" />
</FormItem>
<Row>
{waitTime ? (
<Button type="primary" style={{ width: '100%' }} disabled>
<Countdown
valueStyle={{
color:
theme === 'vs'
? 'rgba(0,0,0,.25)'
: 'rgba(232, 230, 227, 0.25)',
}}
className="inline-countdown"
onFinish={() => setWaitTime(null)}
format="ss"
value={Date.now() + 1000 * waitTime}
/>
</Button>
) : (
<Button
type="primary"
htmlType="submit"
style={{ width: '100%' }}
loading={loading}
>
</Button>
)}
</Row>
</Form>
)}
</div>
<div className={styles.extra}>
{twoFactor ? (
<div style={{ paddingLeft: 20, position: 'relative' }}>
<MobileOutlined
style={{ position: 'absolute', left: 0, top: 4 }}
/>
</div>
) : (
''
)}
</div>
</div>
</div>

View File

@ -22,13 +22,13 @@ import {
auto as followSystemColorScheme,
setFetchMethod,
} from 'darkreader';
import { history } from 'umi';
import AppModal from './appModal';
import {
EditOutlined,
DeleteOutlined,
ReloadOutlined,
} from '@ant-design/icons';
import SecuritySettings from './security';
const { Text } = Typography;
const optionsWithDisabled = [
@ -37,7 +37,7 @@ const optionsWithDisabled = [
{ label: '跟随系统', value: 'auto' },
];
const Setting = ({ headerStyle, isPhone }: any) => {
const Setting = ({ headerStyle, isPhone, user }: any) => {
const columns = [
{
title: '名称',
@ -109,24 +109,7 @@ const Setting = ({ headerStyle, isPhone }: any) => {
const [dataSource, setDataSource] = useState<any[]>([]);
const [isModalVisible, setIsModalVisible] = useState(false);
const [editedApp, setEditedApp] = useState();
const [tabActiveKey, setTabActiveKey] = useState('person');
const handleOk = (values: any) => {
request
.post(`${config.apiPrefix}user`, {
data: {
username: values.username,
password: values.password,
},
})
.then((data: any) => {
localStorage.removeItem(config.authKey);
history.push('/login');
})
.catch((error: any) => {
console.log(error);
});
};
const [tabActiveKey, setTabActiveKey] = useState('security');
const themeChange = (e: any) => {
setTheme(e.target.value);
@ -272,35 +255,13 @@ const Setting = ({ headerStyle, isPhone }: any) => {
}
>
<Tabs
defaultActiveKey="person"
defaultActiveKey="security"
size="small"
tabPosition="top"
onChange={tabChange}
>
<Tabs.TabPane tab="个人设置" key="person">
<Form onFinish={handleOk} layout="vertical">
<Form.Item
label="用户名"
name="username"
rules={[{ required: true }]}
hasFeedback
style={{ maxWidth: 300 }}
>
<Input placeholder="用户名" />
</Form.Item>
<Form.Item
label="密码"
name="password"
rules={[{ required: true }]}
hasFeedback
style={{ maxWidth: 300 }}
>
<Input type="password" placeholder="密码" />
</Form.Item>
<Button type="primary" htmlType="submit">
</Button>
</Form>
<Tabs.TabPane tab="安全设置" key="security">
<SecuritySettings user={user} />
</Tabs.TabPane>
<Tabs.TabPane tab="应用设置" key="app">
<Table

View File

@ -0,0 +1,199 @@
import React, { useEffect, useState } from 'react';
import { Typography, Input, Form, Button, Spin, message } from 'antd';
import { request } from '@/utils/http';
import config from '@/utils/config';
import { history } from 'umi';
import QRCode from 'qrcode.react';
const { Title, Link } = Typography;
const SecuritySettings = ({ user }: any) => {
const [loading, setLoading] = useState(false);
const [twoFactorActived, setTwoFactorActived] = useState<boolean>();
const [twoFactoring, setTwoFactoring] = useState(false);
const [twoFactorInfo, setTwoFactorInfo] = useState<any>();
const [code, setCode] = useState<string>();
const handleOk = (values: any) => {
request
.post(`${config.apiPrefix}user`, {
data: {
username: values.username,
password: values.password,
},
})
.then((data: any) => {
localStorage.removeItem(config.authKey);
history.push('/login');
})
.catch((error: any) => {
console.log(error);
});
};
const activeOrDeactiveTwoFactor = () => {
if (twoFactorActived) {
deactiveTowFactor();
} else {
getTwoFactorInfo();
setTwoFactoring(true);
}
};
const deactiveTowFactor = () => {
request
.put(`${config.apiPrefix}user/two-factor/deactive`)
.then((data: any) => {
if (data.data) {
setTwoFactorActived(false);
}
})
.catch((error: any) => {
console.log(error);
});
};
const completeTowFactor = () => {
request
.put(`${config.apiPrefix}user/two-factor/active`, { data: { code } })
.then((data: any) => {
if (data.data) {
message.success('激活成功');
setTwoFactoring(false);
setTwoFactorActived(true);
}
})
.catch((error: any) => {
console.log(error);
});
};
const getTwoFactorInfo = () => {
request
.get(`${config.apiPrefix}user/two-factor/init`)
.then((data: any) => {
setTwoFactorInfo(data.data);
})
.catch((error: any) => {
console.log(error);
});
};
useEffect(() => {
setTwoFactorActived(user && user.twoFactorActived);
}, [user]);
return twoFactoring ? (
<>
{twoFactorInfo ? (
<div>
<Title level={5}></Title>
Google Authenticator
<Link
href="https://www.microsoft.com/en-us/security/mobile-authenticator-app"
target="_blank"
>
Microsoft Authenticator
</Link>
<Link href="https://authy.com/download/" target="_blank">
Authy
</Link>
<Link
href="https://support.1password.com/one-time-passwords/"
target="_blank"
>
1Password
</Link>
<Link
href="https://support.logmeininc.com/lastpass/help/lastpass-authenticator-lp030014"
target="_blank"
>
LastPass Authenticator
</Link>
<Title style={{ marginTop: 5 }} level={5}>
</Title>
使 {twoFactorInfo?.secret}
<div style={{ marginTop: 10 }}>
<QRCode value={twoFactorInfo?.url} />
</div>
<Title style={{ marginTop: 5 }} level={5}>
</Title>
6
<Input
style={{ margin: '10px 0 10px 0', display: 'block', maxWidth: 200 }}
value={code}
onChange={(e) => setCode(e.target.value)}
placeholder="123456"
/>
<Button type="primary" onClick={completeTowFactor}>
</Button>
</div>
) : (
<Spin />
)}
</>
) : (
<>
<div
style={{
fontSize: 18,
borderBottom: '1px solid #f0f0f0',
marginBottom: 8,
paddingBottom: 4,
}}
>
</div>
<Form onFinish={handleOk} layout="vertical">
<Form.Item
label="用户名"
name="username"
rules={[{ required: true }]}
hasFeedback
style={{ maxWidth: 300 }}
>
<Input placeholder="用户名" />
</Form.Item>
<Form.Item
label="密码"
name="password"
rules={[{ required: true }]}
hasFeedback
style={{ maxWidth: 300 }}
>
<Input type="password" placeholder="密码" />
</Form.Item>
<Button type="primary" htmlType="submit">
</Button>
</Form>
<div
style={{
fontSize: 18,
borderBottom: '1px solid #f0f0f0',
marginBottom: 8,
paddingBottom: 4,
marginTop: 16,
}}
>
</div>
<Button
type="primary"
danger={twoFactorActived}
onClick={activeOrDeactiveTwoFactor}
>
{twoFactorActived ? '禁用' : '启用'}
</Button>
</>
);
};
export default SecuritySettings;