feat: totp
This commit is contained in:
parent
00f26bdc75
commit
3175911105
17 changed files with 615 additions and 32 deletions
|
@ -55,7 +55,9 @@
|
||||||
"ms": "canary",
|
"ms": "canary",
|
||||||
"multer": "^1.4.5-lts.1",
|
"multer": "^1.4.5-lts.1",
|
||||||
"next": "^13.0.0",
|
"next": "^13.0.0",
|
||||||
|
"otplib": "^12.0.1",
|
||||||
"prisma": "^4.5.0",
|
"prisma": "^4.5.0",
|
||||||
|
"qrcode": "^1.5.1",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-chartjs-2": "^4.3.1",
|
"react-chartjs-2": "^4.3.1",
|
||||||
"react-dom": "^18.2.0",
|
"react-dom": "^18.2.0",
|
||||||
|
@ -68,6 +70,7 @@
|
||||||
"@types/minio": "^7.0.14",
|
"@types/minio": "^7.0.14",
|
||||||
"@types/multer": "^1.4.7",
|
"@types/multer": "^1.4.7",
|
||||||
"@types/node": "^18.11.7",
|
"@types/node": "^18.11.7",
|
||||||
|
"@types/qrcode": "^1.5.0",
|
||||||
"@types/react": "^18.0.24",
|
"@types/react": "^18.0.24",
|
||||||
"@types/sharp": "^0.31.0",
|
"@types/sharp": "^0.31.0",
|
||||||
"cross-env": "^7.0.3",
|
"cross-env": "^7.0.3",
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "User" ADD COLUMN "totpSecret" TEXT;
|
|
@ -20,6 +20,7 @@ model User {
|
||||||
embedColor String @default("#2f3136")
|
embedColor String @default("#2f3136")
|
||||||
embedSiteName String? @default("{image.file} • {user.name}")
|
embedSiteName String? @default("{image.file} • {user.name}")
|
||||||
ratelimit DateTime?
|
ratelimit DateTime?
|
||||||
|
totpSecret String?
|
||||||
domains String[]
|
domains String[]
|
||||||
oauth OAuth[]
|
oauth OAuth[]
|
||||||
images Image[]
|
images Image[]
|
||||||
|
|
141
src/components/pages/Manage/TotpModal.tsx
Normal file
141
src/components/pages/Manage/TotpModal.tsx
Normal file
|
@ -0,0 +1,141 @@
|
||||||
|
import { Button, Center, Image, Modal, NumberInput, Text, Title } from '@mantine/core';
|
||||||
|
import { showNotification } from '@mantine/notifications';
|
||||||
|
import { CheckIcon, CrossIcon } from 'components/icons';
|
||||||
|
import useFetch from 'hooks/useFetch';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
export function TotpModal({ opened, onClose, deleteTotp, setTotpEnabled }) {
|
||||||
|
const [secret, setSecret] = useState('');
|
||||||
|
const [qrCode, setQrCode] = useState('');
|
||||||
|
const [disabled, setDisabled] = useState(false);
|
||||||
|
const [code, setCode] = useState(undefined);
|
||||||
|
const [error, setError] = useState('');
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
(async () => {
|
||||||
|
if (opened && !deleteTotp) {
|
||||||
|
const data = await useFetch('/api/user/mfa/totp');
|
||||||
|
if (!data.data_url) {
|
||||||
|
onClose();
|
||||||
|
showNotification({
|
||||||
|
title: 'Error',
|
||||||
|
message: "Can't generate code as you are already using MFA",
|
||||||
|
color: 'red',
|
||||||
|
icon: <CrossIcon />,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setSecret(data.secret);
|
||||||
|
setQrCode(data.data_url);
|
||||||
|
setError('');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
}, [opened]);
|
||||||
|
|
||||||
|
const disableTotp = async () => {
|
||||||
|
setDisabled(true);
|
||||||
|
const str = code.toString();
|
||||||
|
if (str.length !== 6) {
|
||||||
|
return setError('Code must be 6 digits');
|
||||||
|
}
|
||||||
|
|
||||||
|
const resp = await useFetch('/api/user/mfa/totp', 'DELETE', {
|
||||||
|
code: str,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (resp.error) {
|
||||||
|
setError(resp.error);
|
||||||
|
} else {
|
||||||
|
showNotification({
|
||||||
|
title: 'Success',
|
||||||
|
message: 'Successfully disabled MFA',
|
||||||
|
color: 'green',
|
||||||
|
icon: <CheckIcon />,
|
||||||
|
});
|
||||||
|
|
||||||
|
setTotpEnabled(false);
|
||||||
|
|
||||||
|
onClose();
|
||||||
|
}
|
||||||
|
|
||||||
|
setDisabled(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const verifyCode = async () => {
|
||||||
|
setDisabled(true);
|
||||||
|
const str = code.toString();
|
||||||
|
if (str.length !== 6) {
|
||||||
|
return setError('Code must be 6 digits');
|
||||||
|
}
|
||||||
|
|
||||||
|
const resp = await useFetch('/api/user/mfa/totp', 'POST', {
|
||||||
|
secret,
|
||||||
|
code: str,
|
||||||
|
register: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (resp.error) {
|
||||||
|
setError(resp.error);
|
||||||
|
} else {
|
||||||
|
showNotification({
|
||||||
|
title: 'Success',
|
||||||
|
message: 'Successfully enabled MFA',
|
||||||
|
color: 'green',
|
||||||
|
icon: <CheckIcon />,
|
||||||
|
});
|
||||||
|
|
||||||
|
setTotpEnabled(true);
|
||||||
|
|
||||||
|
onClose();
|
||||||
|
}
|
||||||
|
|
||||||
|
setDisabled(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
opened={opened}
|
||||||
|
onClose={onClose}
|
||||||
|
title={<Title order={3}>Two-Factor Authentication</Title>}
|
||||||
|
size='lg'
|
||||||
|
>
|
||||||
|
{deleteTotp ? (
|
||||||
|
<Text mb='md'>Verify your code to disable Two-Factor Authentication</Text>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<Text mb='md'>
|
||||||
|
Scan the QR Code below in <b>Authy</b>, <b>Google Authenticator</b>, or any other supported
|
||||||
|
client.
|
||||||
|
</Text>
|
||||||
|
<Center>
|
||||||
|
<Image height={180} width={180} src={qrCode} alt='QR Code' withPlaceholder />
|
||||||
|
</Center>
|
||||||
|
<Text my='sm'>QR Code not working? Try manually entering the code into your app: {secret}</Text>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<NumberInput
|
||||||
|
placeholder='2FA Code'
|
||||||
|
label='Verify'
|
||||||
|
size='xl'
|
||||||
|
hideControls
|
||||||
|
maxLength={6}
|
||||||
|
minLength={6}
|
||||||
|
value={code}
|
||||||
|
onChange={(e) => setCode(e)}
|
||||||
|
error={error}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
disabled={disabled}
|
||||||
|
size='lg'
|
||||||
|
fullWidth
|
||||||
|
mt='md'
|
||||||
|
rightIcon={<CheckIcon />}
|
||||||
|
onClick={deleteTotp ? disableTotp : verifyCode}
|
||||||
|
>
|
||||||
|
Verify{deleteTotp ? ' and Disable' : ''}
|
||||||
|
</Button>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
}
|
|
@ -43,6 +43,7 @@ import { useEffect, useState } from 'react';
|
||||||
import { useRecoilState } from 'recoil';
|
import { useRecoilState } from 'recoil';
|
||||||
import Flameshot from './Flameshot';
|
import Flameshot from './Flameshot';
|
||||||
import ShareX from './ShareX';
|
import ShareX from './ShareX';
|
||||||
|
import { TotpModal } from './TotpModal';
|
||||||
|
|
||||||
function ExportDataTooltip({ children }) {
|
function ExportDataTooltip({ children }) {
|
||||||
return (
|
return (
|
||||||
|
@ -56,7 +57,7 @@ function ExportDataTooltip({ children }) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Manage({ oauth_registration, oauth_providers: raw_oauth_providers }) {
|
export default function Manage({ oauth_registration, oauth_providers: raw_oauth_providers, totp_enabled }) {
|
||||||
const oauth_providers = JSON.parse(raw_oauth_providers);
|
const oauth_providers = JSON.parse(raw_oauth_providers);
|
||||||
const icons = {
|
const icons = {
|
||||||
Discord: DiscordIcon,
|
Discord: DiscordIcon,
|
||||||
|
@ -71,11 +72,13 @@ export default function Manage({ oauth_registration, oauth_providers: raw_oauth_
|
||||||
const [user, setUser] = useRecoilState(userSelector);
|
const [user, setUser] = useRecoilState(userSelector);
|
||||||
const modals = useModals();
|
const modals = useModals();
|
||||||
|
|
||||||
|
const [totpOpen, setTotpOpen] = useState(false);
|
||||||
const [shareXOpen, setShareXOpen] = useState(false);
|
const [shareXOpen, setShareXOpen] = useState(false);
|
||||||
const [flameshotOpen, setFlameshotOpen] = useState(false);
|
const [flameshotOpen, setFlameshotOpen] = useState(false);
|
||||||
const [exports, setExports] = useState([]);
|
const [exports, setExports] = useState([]);
|
||||||
const [file, setFile] = useState<File>(null);
|
const [file, setFile] = useState<File>(null);
|
||||||
const [fileDataURL, setFileDataURL] = useState(user.avatar ?? null);
|
const [fileDataURL, setFileDataURL] = useState(user.avatar ?? null);
|
||||||
|
const [totpEnabled, setTotpEnabled] = useState(!!user.totpSecret);
|
||||||
|
|
||||||
const getDataURL = (f: File): Promise<string> => {
|
const getDataURL = (f: File): Promise<string> => {
|
||||||
return new Promise((res, rej) => {
|
return new Promise((res, rej) => {
|
||||||
|
@ -372,6 +375,28 @@ export default function Manage({ oauth_registration, oauth_providers: raw_oauth_
|
||||||
</Group>
|
</Group>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
|
{totp_enabled && (
|
||||||
|
<Box my='md'>
|
||||||
|
<Title>Two Factor Authentication</Title>
|
||||||
|
<MutedText size='md'>
|
||||||
|
{user.totpSecret
|
||||||
|
? 'You have two factor authentication enabled.'
|
||||||
|
: 'You do not have two factor authentication enabled.'}
|
||||||
|
</MutedText>
|
||||||
|
|
||||||
|
<Button size='lg' my='sm' onClick={() => setTotpOpen(true)}>
|
||||||
|
{totpEnabled ? 'Disable' : 'Enable'} Two Factor Authentication
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<TotpModal
|
||||||
|
opened={totpOpen}
|
||||||
|
onClose={() => setTotpOpen(false)}
|
||||||
|
deleteTotp={totpEnabled}
|
||||||
|
setTotpEnabled={setTotpEnabled}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
)}
|
||||||
|
|
||||||
{oauth_registration && (
|
{oauth_registration && (
|
||||||
<Box my='md'>
|
<Box my='md'>
|
||||||
<Title>OAuth</Title>
|
<Title>OAuth</Title>
|
||||||
|
|
|
@ -123,6 +123,11 @@ export interface ConfigChunks {
|
||||||
chunks_size: number;
|
chunks_size: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface ConfigMfa {
|
||||||
|
totp_enabled: boolean;
|
||||||
|
totp_issuer: string;
|
||||||
|
}
|
||||||
|
|
||||||
export interface Config {
|
export interface Config {
|
||||||
core: ConfigCore;
|
core: ConfigCore;
|
||||||
uploader: ConfigUploader;
|
uploader: ConfigUploader;
|
||||||
|
@ -134,4 +139,5 @@ export interface Config {
|
||||||
oauth: ConfigOAuth;
|
oauth: ConfigOAuth;
|
||||||
features: ConfigFeatures;
|
features: ConfigFeatures;
|
||||||
chunks: ConfigChunks;
|
chunks: ConfigChunks;
|
||||||
|
mfa: ConfigMfa;
|
||||||
}
|
}
|
||||||
|
|
|
@ -143,6 +143,9 @@ export default function readConfig() {
|
||||||
|
|
||||||
map('CHUNKS_MAX_SIZE', 'human-to-byte', 'chunks.max_size'),
|
map('CHUNKS_MAX_SIZE', 'human-to-byte', 'chunks.max_size'),
|
||||||
map('CHUNKS_CHUNKS_SIZE', 'human-to-byte', 'chunks.chunks_size'),
|
map('CHUNKS_CHUNKS_SIZE', 'human-to-byte', 'chunks.chunks_size'),
|
||||||
|
|
||||||
|
map('MFA_TOTP_ISSUER', 'string', 'mfa.totp_issuer'),
|
||||||
|
map('MFA_TOTP_ENABLED', 'boolean', 'mfa.totp_enabled'),
|
||||||
];
|
];
|
||||||
|
|
||||||
const config = {};
|
const config = {};
|
||||||
|
|
|
@ -172,7 +172,12 @@ const validator = s.object({
|
||||||
oauth_registration: s.boolean.default(false),
|
oauth_registration: s.boolean.default(false),
|
||||||
user_registration: s.boolean.default(false),
|
user_registration: s.boolean.default(false),
|
||||||
})
|
})
|
||||||
.default({ invites: false, invites_length: 6, oauth_registration: false, user_registration: false }),
|
.default({
|
||||||
|
invites: false,
|
||||||
|
invites_length: 6,
|
||||||
|
oauth_registration: false,
|
||||||
|
user_registration: false,
|
||||||
|
}),
|
||||||
chunks: s
|
chunks: s
|
||||||
.object({
|
.object({
|
||||||
max_size: s.number.default(humanToBytes('90MB')),
|
max_size: s.number.default(humanToBytes('90MB')),
|
||||||
|
@ -182,6 +187,15 @@ const validator = s.object({
|
||||||
max_size: humanToBytes('90MB'),
|
max_size: humanToBytes('90MB'),
|
||||||
chunks_size: humanToBytes('20MB'),
|
chunks_size: humanToBytes('20MB'),
|
||||||
}),
|
}),
|
||||||
|
mfa: s
|
||||||
|
.object({
|
||||||
|
totp_issuer: s.string.default('Zipline'),
|
||||||
|
totp_enabled: s.boolean.default(false),
|
||||||
|
})
|
||||||
|
.default({
|
||||||
|
totp_issuer: 'Zipline',
|
||||||
|
totp_enabled: false,
|
||||||
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
export default function validate(config): Config {
|
export default function validate(config): Config {
|
||||||
|
@ -224,6 +238,8 @@ export default function validate(config): Config {
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (process.env.ZIPLINE_DOCKER_BUILD) return null;
|
if (process.env.ZIPLINE_DOCKER_BUILD) return null;
|
||||||
|
|
||||||
|
logger.debug(`config error: ${inspect(e, { depth: Infinity })}`);
|
||||||
|
|
||||||
e.stack = '';
|
e.stack = '';
|
||||||
|
|
||||||
Logger.get('config')
|
Logger.get('config')
|
||||||
|
|
|
@ -2,13 +2,31 @@ import config from 'lib/config';
|
||||||
import { notNull } from 'lib/util';
|
import { notNull } from 'lib/util';
|
||||||
import { GetServerSideProps } from 'next';
|
import { GetServerSideProps } from 'next';
|
||||||
|
|
||||||
export const getServerSideProps: GetServerSideProps = async (ctx) => {
|
export type OauthProvider = {
|
||||||
// this entire thing will also probably change before the stable release
|
name: string;
|
||||||
|
url: string;
|
||||||
|
link_url: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ServerSideProps = {
|
||||||
|
title: string;
|
||||||
|
external_links: string;
|
||||||
|
disable_media_preview: boolean;
|
||||||
|
invites: boolean;
|
||||||
|
user_registration: boolean;
|
||||||
|
oauth_registration: boolean;
|
||||||
|
oauth_providers: string;
|
||||||
|
chunks_size: number;
|
||||||
|
max_size: number;
|
||||||
|
totp_enabled: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getServerSideProps: GetServerSideProps<ServerSideProps> = async () => {
|
||||||
const ghEnabled = notNull(config.oauth?.github_client_id, config.oauth?.github_client_secret);
|
const ghEnabled = notNull(config.oauth?.github_client_id, config.oauth?.github_client_secret);
|
||||||
const discEnabled = notNull(config.oauth?.discord_client_id, config.oauth?.discord_client_secret);
|
const discEnabled = notNull(config.oauth?.discord_client_id, config.oauth?.discord_client_secret);
|
||||||
const googleEnabled = notNull(config.oauth?.google_client_id, config.oauth?.google_client_secret);
|
const googleEnabled = notNull(config.oauth?.google_client_id, config.oauth?.google_client_secret);
|
||||||
|
|
||||||
const oauth_providers = [];
|
const oauth_providers: OauthProvider[] = [];
|
||||||
|
|
||||||
if (ghEnabled)
|
if (ghEnabled)
|
||||||
oauth_providers.push({
|
oauth_providers.push({
|
||||||
|
@ -41,6 +59,7 @@ export const getServerSideProps: GetServerSideProps = async (ctx) => {
|
||||||
oauth_providers: JSON.stringify(oauth_providers),
|
oauth_providers: JSON.stringify(oauth_providers),
|
||||||
chunks_size: config.chunks.chunks_size,
|
chunks_size: config.chunks.chunks_size,
|
||||||
max_size: config.chunks.max_size,
|
max_size: config.chunks.max_size,
|
||||||
|
totp_enabled: config.mfa.totp_enabled,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
|
@ -7,6 +7,7 @@ import { HTTPMethod } from 'find-my-way';
|
||||||
import config from 'lib/config';
|
import config from 'lib/config';
|
||||||
import prisma from 'lib/prisma';
|
import prisma from 'lib/prisma';
|
||||||
import { sign64, unsign64 } from 'lib/utils/crypto';
|
import { sign64, unsign64 } from 'lib/utils/crypto';
|
||||||
|
import Logger from 'lib/logger';
|
||||||
|
|
||||||
export interface NextApiFile {
|
export interface NextApiFile {
|
||||||
fieldname: string;
|
fieldname: string;
|
||||||
|
@ -180,6 +181,8 @@ export const withZipline =
|
||||||
|
|
||||||
const signed = sign64(String(value), config.core.secret);
|
const signed = sign64(String(value), config.core.secret);
|
||||||
|
|
||||||
|
Logger.get('api').debug(`headers(${JSON.stringify(req.headers)}): cookie(${name}, ${value})`);
|
||||||
|
|
||||||
res.setHeader('Set-Cookie', serialize(name, signed, options));
|
res.setHeader('Set-Cookie', serialize(name, signed, options));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,27 +1,12 @@
|
||||||
import { OAuth } from '@prisma/client';
|
import type { UserExtended } from 'middleware/withZipline';
|
||||||
import { atom, selector } from 'recoil';
|
import { atom, selector } from 'recoil';
|
||||||
|
|
||||||
export interface User {
|
|
||||||
username: string;
|
|
||||||
token: string;
|
|
||||||
embedTitle: string;
|
|
||||||
embedColor: string;
|
|
||||||
embedSiteName: string;
|
|
||||||
systemTheme: string;
|
|
||||||
domains: string[];
|
|
||||||
avatar?: string;
|
|
||||||
administrator: boolean;
|
|
||||||
superAdmin: boolean;
|
|
||||||
oauth: OAuth[];
|
|
||||||
id: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const userState = atom({
|
export const userState = atom({
|
||||||
key: 'userState',
|
key: 'userState',
|
||||||
default: null as User,
|
default: null as UserExtended,
|
||||||
});
|
});
|
||||||
|
|
||||||
export const userSelector = selector<User>({
|
export const userSelector = selector<UserExtended>({
|
||||||
key: 'userSelector',
|
key: 'userSelector',
|
||||||
get: ({ get }) => get(userState),
|
get: ({ get }) => get(userState),
|
||||||
set: ({ set }, newValue) => set(userState, newValue),
|
set: ({ set }, newValue) => set(userState, newValue),
|
||||||
|
|
16
src/lib/utils/totp.ts
Normal file
16
src/lib/utils/totp.ts
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
import { authenticator } from 'otplib';
|
||||||
|
import { toDataURL } from 'qrcode';
|
||||||
|
|
||||||
|
export function generate_totp_secret() {
|
||||||
|
return authenticator.generateSecret(32);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function verify_totp_code(secret: string, code: string) {
|
||||||
|
return authenticator.check(code, secret);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function totp_qrcode(issuer: string, username: string, secret: string): Promise<string> {
|
||||||
|
const url = authenticator.keyuri(username, issuer, secret);
|
||||||
|
|
||||||
|
return toDataURL(url);
|
||||||
|
}
|
|
@ -1,14 +1,17 @@
|
||||||
import prisma from 'lib/prisma';
|
import config from 'lib/config';
|
||||||
import { NextApiReq, NextApiRes, withZipline } from 'middleware/withZipline';
|
|
||||||
import { checkPassword, createToken, hashPassword } from 'lib/util';
|
|
||||||
import Logger from 'lib/logger';
|
import Logger from 'lib/logger';
|
||||||
|
import prisma from 'lib/prisma';
|
||||||
|
import { checkPassword, createToken, hashPassword } from 'lib/util';
|
||||||
|
import { verify_totp_code } from 'lib/utils/totp';
|
||||||
|
import { NextApiReq, NextApiRes, withZipline } from 'middleware/withZipline';
|
||||||
|
|
||||||
async function handler(req: NextApiReq, res: NextApiRes) {
|
async function handler(req: NextApiReq, res: NextApiRes) {
|
||||||
const logger = Logger.get('login');
|
const logger = Logger.get('login');
|
||||||
|
|
||||||
const { username, password } = req.body as {
|
const { username, password, code } = req.body as {
|
||||||
username: string;
|
username: string;
|
||||||
password: string;
|
password: string;
|
||||||
|
code?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
const users = await prisma.user.findMany();
|
const users = await prisma.user.findMany();
|
||||||
|
@ -33,9 +36,25 @@ async function handler(req: NextApiReq, res: NextApiRes) {
|
||||||
|
|
||||||
if (!user) return res.notFound('user not found');
|
if (!user) return res.notFound('user not found');
|
||||||
|
|
||||||
const valid = await checkPassword(password, user.password);
|
let valid = false;
|
||||||
|
if (user.token === password) valid = true;
|
||||||
|
else if (await checkPassword(password, user.password)) valid = true;
|
||||||
|
else valid = false;
|
||||||
|
|
||||||
|
logger.debug(`body(${JSON.stringify(req.body)}): checkPassword(${password}, argon2-str) => ${valid}`);
|
||||||
|
|
||||||
if (!valid) return res.unauthorized('Wrong password');
|
if (!valid) return res.unauthorized('Wrong password');
|
||||||
|
|
||||||
|
if (user.totpSecret && config.mfa.totp_enabled) {
|
||||||
|
if (!code) return res.unauthorized('TOTP required', { totp: true });
|
||||||
|
|
||||||
|
const success = verify_totp_code(user.totpSecret, code);
|
||||||
|
logger.debug(
|
||||||
|
`body(${JSON.stringify(req.body)}): verify_totp_code(${user.totpSecret}, ${code}) => ${success}`
|
||||||
|
);
|
||||||
|
if (!success) return res.badRequest('Invalid code', { totp: true });
|
||||||
|
}
|
||||||
|
|
||||||
res.setUserCookie(user.id);
|
res.setUserCookie(user.id);
|
||||||
logger.info(`User ${user.username} (${user.id}) logged in`);
|
logger.info(`User ${user.username} (${user.id}) logged in`);
|
||||||
|
|
||||||
|
|
82
src/pages/api/user/mfa/totp.ts
Normal file
82
src/pages/api/user/mfa/totp.ts
Normal file
|
@ -0,0 +1,82 @@
|
||||||
|
import config from 'lib/config';
|
||||||
|
import Logger from 'lib/logger';
|
||||||
|
import prisma from 'lib/prisma';
|
||||||
|
import { generate_totp_secret, totp_qrcode, verify_totp_code } from 'lib/utils/totp';
|
||||||
|
import { NextApiReq, NextApiRes, UserExtended, withZipline } from 'middleware/withZipline';
|
||||||
|
|
||||||
|
const logger = Logger.get('user::mfa::totp');
|
||||||
|
|
||||||
|
async function handler(req: NextApiReq, res: NextApiRes, user: UserExtended) {
|
||||||
|
if (!config.mfa.totp_enabled) return res.forbidden('totp is disabled');
|
||||||
|
|
||||||
|
if (req.method === 'POST') {
|
||||||
|
const { secret, code } = req.body as { secret: string; code: string };
|
||||||
|
|
||||||
|
if (!secret) return res.badRequest('no secret');
|
||||||
|
if (!code) return res.badRequest('no code');
|
||||||
|
|
||||||
|
if (code.length !== 6) return res.badRequest('invalid code (code.length != 6)');
|
||||||
|
|
||||||
|
const success = verify_totp_code(secret, code);
|
||||||
|
logger.debug(`body(${JSON.stringify(req.body)}): verify_totp_code(${secret}, ${code}) => ${success}`);
|
||||||
|
|
||||||
|
if (!success) return res.badRequest('Invalid code');
|
||||||
|
if (user.totpSecret) return res.badRequest('totp already registered');
|
||||||
|
|
||||||
|
logger.debug(`registering totp(${secret}) ${user.id}`);
|
||||||
|
await prisma.user.update({
|
||||||
|
where: { id: user.id },
|
||||||
|
data: {
|
||||||
|
totpSecret: secret,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
delete user.password;
|
||||||
|
return res.json(user);
|
||||||
|
} else if (req.method === 'DELETE') {
|
||||||
|
const { code } = req.body as { code: string };
|
||||||
|
|
||||||
|
if (!code) return res.badRequest('no code');
|
||||||
|
if (code.length !== 6) return res.badRequest('invalid code (code.length != 6)');
|
||||||
|
|
||||||
|
const success = verify_totp_code(user.totpSecret, req.body.code);
|
||||||
|
|
||||||
|
logger.debug(
|
||||||
|
`body(${JSON.stringify(req.body)}): verify_totp_code(${user.totpSecret}, ${
|
||||||
|
req.body.code
|
||||||
|
}) => ${success}`
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!success) return res.badRequest('Invalid code');
|
||||||
|
|
||||||
|
logger.debug(`unregistering totp ${user.id}`);
|
||||||
|
await prisma.user.update({
|
||||||
|
where: { id: user.id },
|
||||||
|
data: {
|
||||||
|
totpSecret: null,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
delete user.password;
|
||||||
|
return res.json(user);
|
||||||
|
} else {
|
||||||
|
if (!user.totpSecret) {
|
||||||
|
const secret = generate_totp_secret();
|
||||||
|
const data_url = await totp_qrcode(config.mfa.totp_issuer, user.username, secret);
|
||||||
|
|
||||||
|
return res.json({
|
||||||
|
secret,
|
||||||
|
data_url,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.json({
|
||||||
|
secret: user.totpSecret,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default withZipline(handler, {
|
||||||
|
methods: ['GET', 'POST', 'DELETE'],
|
||||||
|
user: true,
|
||||||
|
});
|
|
@ -1,16 +1,32 @@
|
||||||
import { Button, Center, Divider, PasswordInput, TextInput, Title } from '@mantine/core';
|
import {
|
||||||
|
Button,
|
||||||
|
Center,
|
||||||
|
CheckIcon,
|
||||||
|
Divider,
|
||||||
|
Modal,
|
||||||
|
NumberInput,
|
||||||
|
PasswordInput,
|
||||||
|
TextInput,
|
||||||
|
Title,
|
||||||
|
} from '@mantine/core';
|
||||||
import { useForm } from '@mantine/form';
|
import { useForm } from '@mantine/form';
|
||||||
import { DiscordIcon, GitHubIcon, GoogleIcon } from 'components/icons';
|
import { DiscordIcon, GitHubIcon, GoogleIcon } from 'components/icons';
|
||||||
import useFetch from 'hooks/useFetch';
|
import useFetch from 'hooks/useFetch';
|
||||||
import Head from 'next/head';
|
import Head from 'next/head';
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import { useRouter } from 'next/router';
|
import { useRouter } from 'next/router';
|
||||||
import { useEffect } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
export { getServerSideProps } from 'middleware/getServerSideProps';
|
export { getServerSideProps } from 'middleware/getServerSideProps';
|
||||||
|
|
||||||
export default function Login({ title, user_registration, oauth_registration, oauth_providers: unparsed }) {
|
export default function Login({ title, user_registration, oauth_registration, oauth_providers: unparsed }) {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
|
// totp modal
|
||||||
|
const [totpOpen, setTotpOpen] = useState(false);
|
||||||
|
const [code, setCode] = useState(undefined);
|
||||||
|
const [error, setError] = useState('');
|
||||||
|
const [disabled, setDisabled] = useState(false);
|
||||||
|
|
||||||
const oauth_providers = JSON.parse(unparsed);
|
const oauth_providers = JSON.parse(unparsed);
|
||||||
|
|
||||||
const icons = {
|
const icons = {
|
||||||
|
@ -31,6 +47,7 @@ export default function Login({ title, user_registration, oauth_registration, oa
|
||||||
});
|
});
|
||||||
|
|
||||||
const onSubmit = async (values) => {
|
const onSubmit = async (values) => {
|
||||||
|
setError('');
|
||||||
const username = values.username.trim();
|
const username = values.username.trim();
|
||||||
const password = values.password.trim();
|
const password = values.password.trim();
|
||||||
|
|
||||||
|
@ -39,11 +56,20 @@ export default function Login({ title, user_registration, oauth_registration, oa
|
||||||
const res = await useFetch('/api/auth/login', 'POST', {
|
const res = await useFetch('/api/auth/login', 'POST', {
|
||||||
username,
|
username,
|
||||||
password,
|
password,
|
||||||
|
code: code?.toString() || null,
|
||||||
});
|
});
|
||||||
|
|
||||||
if (res.error) {
|
if (res.error) {
|
||||||
if (res.code === 403) {
|
if (res.code === 403) {
|
||||||
form.setFieldError('password', 'Invalid password');
|
form.setFieldError('password', 'Invalid password');
|
||||||
|
} else if (res.totp) {
|
||||||
|
if (res.code === 400) {
|
||||||
|
setError('Invalid code');
|
||||||
|
} else {
|
||||||
|
setError('');
|
||||||
|
}
|
||||||
|
|
||||||
|
setTotpOpen(true);
|
||||||
} else {
|
} else {
|
||||||
form.setFieldError('username', 'Invalid username');
|
form.setFieldError('username', 'Invalid username');
|
||||||
form.setFieldError('password', 'Invalid password');
|
form.setFieldError('password', 'Invalid password');
|
||||||
|
@ -66,6 +92,35 @@ export default function Login({ title, user_registration, oauth_registration, oa
|
||||||
<Head>
|
<Head>
|
||||||
<title>{full_title}</title>
|
<title>{full_title}</title>
|
||||||
</Head>
|
</Head>
|
||||||
|
<Modal
|
||||||
|
opened={totpOpen}
|
||||||
|
onClose={() => setTotpOpen(false)}
|
||||||
|
title={<Title order={3}>Two-Factor Authentication Required</Title>}
|
||||||
|
size='lg'
|
||||||
|
>
|
||||||
|
<NumberInput
|
||||||
|
placeholder='2FA Code'
|
||||||
|
label='Verify'
|
||||||
|
size='xl'
|
||||||
|
hideControls
|
||||||
|
maxLength={6}
|
||||||
|
minLength={6}
|
||||||
|
value={code}
|
||||||
|
onChange={(e) => setCode(e)}
|
||||||
|
error={error}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
disabled={disabled}
|
||||||
|
size='lg'
|
||||||
|
fullWidth
|
||||||
|
mt='md'
|
||||||
|
rightIcon={<CheckIcon />}
|
||||||
|
onClick={() => onSubmit(form.values)}
|
||||||
|
>
|
||||||
|
Verify & Login
|
||||||
|
</Button>
|
||||||
|
</Modal>
|
||||||
<Center sx={{ height: '100vh' }}>
|
<Center sx={{ height: '100vh' }}>
|
||||||
<div>
|
<div>
|
||||||
<Title size={70} align='center'>
|
<Title size={70} align='center'>
|
||||||
|
|
|
@ -2,10 +2,11 @@ import { LoadingOverlay } from '@mantine/core';
|
||||||
import Layout from 'components/Layout';
|
import Layout from 'components/Layout';
|
||||||
import Manage from 'components/pages/Manage';
|
import Manage from 'components/pages/Manage';
|
||||||
import useLogin from 'hooks/useLogin';
|
import useLogin from 'hooks/useLogin';
|
||||||
|
import type { ServerSideProps } from 'middleware/getServerSideProps';
|
||||||
import Head from 'next/head';
|
import Head from 'next/head';
|
||||||
export { getServerSideProps } from 'middleware/getServerSideProps';
|
export { getServerSideProps } from 'middleware/getServerSideProps';
|
||||||
|
|
||||||
export default function ManagePage(props) {
|
export default function ManagePage(props: ServerSideProps) {
|
||||||
const { loading } = useLogin();
|
const { loading } = useLogin();
|
||||||
|
|
||||||
if (loading) return <LoadingOverlay visible={loading} />;
|
if (loading) return <LoadingOverlay visible={loading} />;
|
||||||
|
@ -17,7 +18,11 @@ export default function ManagePage(props) {
|
||||||
<title>{title}</title>
|
<title>{title}</title>
|
||||||
</Head>
|
</Head>
|
||||||
<Layout props={props}>
|
<Layout props={props}>
|
||||||
<Manage oauth_providers={props.oauth_providers} oauth_registration={props.oauth_registration} />
|
<Manage
|
||||||
|
oauth_providers={props.oauth_providers}
|
||||||
|
oauth_registration={props.oauth_registration}
|
||||||
|
totp_enabled={props.totp_enabled}
|
||||||
|
/>
|
||||||
</Layout>
|
</Layout>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|
202
yarn.lock
202
yarn.lock
|
@ -1090,6 +1090,54 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@otplib/core@npm:^12.0.1":
|
||||||
|
version: 12.0.1
|
||||||
|
resolution: "@otplib/core@npm:12.0.1"
|
||||||
|
checksum: b3c34bc20b31bc3f49cc0dc3c0eb070491c0101e8c1efa83cec48ca94158bd736aaca8187df667fc0c4a239d4ac52076bc44084bee04a50c80c3630caf77affa
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@otplib/plugin-crypto@npm:^12.0.1":
|
||||||
|
version: 12.0.1
|
||||||
|
resolution: "@otplib/plugin-crypto@npm:12.0.1"
|
||||||
|
dependencies:
|
||||||
|
"@otplib/core": ^12.0.1
|
||||||
|
checksum: 6867c74ee8aca6c2db9670362cf51e44f3648602c39318bf537421242e33f0012a172acd43bbed9a21d706e535dc4c66aff965380673391e9fd74cf685b5b13a
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@otplib/plugin-thirty-two@npm:^12.0.1":
|
||||||
|
version: 12.0.1
|
||||||
|
resolution: "@otplib/plugin-thirty-two@npm:12.0.1"
|
||||||
|
dependencies:
|
||||||
|
"@otplib/core": ^12.0.1
|
||||||
|
thirty-two: ^1.0.2
|
||||||
|
checksum: 920099e40d3e8c2941291c84c70064c2d86d0d1ed17230d650445d5463340e406bc413ddf2e40c374ddc4ee988ef1e3facacab9b5248b1ff361fd13df52bf88f
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@otplib/preset-default@npm:^12.0.1":
|
||||||
|
version: 12.0.1
|
||||||
|
resolution: "@otplib/preset-default@npm:12.0.1"
|
||||||
|
dependencies:
|
||||||
|
"@otplib/core": ^12.0.1
|
||||||
|
"@otplib/plugin-crypto": ^12.0.1
|
||||||
|
"@otplib/plugin-thirty-two": ^12.0.1
|
||||||
|
checksum: 8133231384f6277f77eb8e42ef83bc32a8b01059bef147d1c358d9e9bfd292e1c239f581fe008367a48489dd68952b7ac0948e6c41412fc06079da2c91b71d16
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@otplib/preset-v11@npm:^12.0.1":
|
||||||
|
version: 12.0.1
|
||||||
|
resolution: "@otplib/preset-v11@npm:12.0.1"
|
||||||
|
dependencies:
|
||||||
|
"@otplib/core": ^12.0.1
|
||||||
|
"@otplib/plugin-crypto": ^12.0.1
|
||||||
|
"@otplib/plugin-thirty-two": ^12.0.1
|
||||||
|
checksum: 367cb09397e617c21ec748d54e920ab43f1c5dfba70cbfd88edf73aecca399cf0c09fefe32518f79c7ee8a06e7058d14b200da378cc7d46af3cac4e22a153e2f
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@phc/format@npm:^1.0.0":
|
"@phc/format@npm:^1.0.0":
|
||||||
version: 1.0.0
|
version: 1.0.0
|
||||||
resolution: "@phc/format@npm:1.0.0"
|
resolution: "@phc/format@npm:1.0.0"
|
||||||
|
@ -1694,6 +1742,15 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/qrcode@npm:^1.5.0":
|
||||||
|
version: 1.5.0
|
||||||
|
resolution: "@types/qrcode@npm:1.5.0"
|
||||||
|
dependencies:
|
||||||
|
"@types/node": "*"
|
||||||
|
checksum: b0ece3834ca5ba6171132928fd1ef764772dc619b45cb4123461ee05e377ad15553a330d234c69db0d0028c6639a99429e88d99192fbba9c5ee97c23f278c48b
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@types/qs@npm:*":
|
"@types/qs@npm:*":
|
||||||
version: 6.9.7
|
version: 6.9.7
|
||||||
resolution: "@types/qs@npm:6.9.7"
|
resolution: "@types/qs@npm:6.9.7"
|
||||||
|
@ -2467,6 +2524,13 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"camelcase@npm:^5.0.0":
|
||||||
|
version: 5.3.1
|
||||||
|
resolution: "camelcase@npm:5.3.1"
|
||||||
|
checksum: e6effce26b9404e3c0f301498184f243811c30dfe6d0b9051863bd8e4034d09c8c2923794f280d6827e5aa055f6c434115ff97864a16a963366fb35fd673024b
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"caniuse-lite@npm:^1.0.30001406":
|
"caniuse-lite@npm:^1.0.30001406":
|
||||||
version: 1.0.30001423
|
version: 1.0.30001423
|
||||||
resolution: "caniuse-lite@npm:1.0.30001423"
|
resolution: "caniuse-lite@npm:1.0.30001423"
|
||||||
|
@ -2597,6 +2661,17 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"cliui@npm:^6.0.0":
|
||||||
|
version: 6.0.0
|
||||||
|
resolution: "cliui@npm:6.0.0"
|
||||||
|
dependencies:
|
||||||
|
string-width: ^4.2.0
|
||||||
|
strip-ansi: ^6.0.0
|
||||||
|
wrap-ansi: ^6.2.0
|
||||||
|
checksum: 4fcfd26d292c9f00238117f39fc797608292ae36bac2168cfee4c85923817d0607fe21b3329a8621e01aedf512c99b7eaa60e363a671ffd378df6649fb48ae42
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"clone@npm:^1.0.2":
|
"clone@npm:^1.0.2":
|
||||||
version: 1.0.4
|
version: 1.0.4
|
||||||
resolution: "clone@npm:1.0.4"
|
resolution: "clone@npm:1.0.4"
|
||||||
|
@ -2964,6 +3039,13 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"decamelize@npm:^1.2.0":
|
||||||
|
version: 1.2.0
|
||||||
|
resolution: "decamelize@npm:1.2.0"
|
||||||
|
checksum: ad8c51a7e7e0720c70ec2eeb1163b66da03e7616d7b98c9ef43cce2416395e84c1e9548dd94f5f6ffecfee9f8b94251fc57121a8b021f2ff2469b2bae247b8aa
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"decode-uri-component@npm:^0.2.0":
|
"decode-uri-component@npm:^0.2.0":
|
||||||
version: 0.2.0
|
version: 0.2.0
|
||||||
resolution: "decode-uri-component@npm:0.2.0"
|
resolution: "decode-uri-component@npm:0.2.0"
|
||||||
|
@ -3092,6 +3174,13 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"dijkstrajs@npm:^1.0.1":
|
||||||
|
version: 1.0.2
|
||||||
|
resolution: "dijkstrajs@npm:1.0.2"
|
||||||
|
checksum: 8cd822441a26f190da24d69bfab7b433d080b09e069e41e046ac84e152f182a1ed9478d531b34126e000adaa7b73114a0f85fcac117a7d25b3edf302d57c0d09
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"dir-glob@npm:^3.0.1":
|
"dir-glob@npm:^3.0.1":
|
||||||
version: 3.0.1
|
version: 3.0.1
|
||||||
resolution: "dir-glob@npm:3.0.1"
|
resolution: "dir-glob@npm:3.0.1"
|
||||||
|
@ -3235,6 +3324,13 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"encode-utf8@npm:^1.0.3":
|
||||||
|
version: 1.0.3
|
||||||
|
resolution: "encode-utf8@npm:1.0.3"
|
||||||
|
checksum: 550224bf2a104b1d355458c8a82e9b4ea07f9fc78387bc3a49c151b940ad26473de8dc9e121eefc4e84561cb0b46de1e4cd2bc766f72ee145e9ea9541482817f
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"encoding@npm:^0.1.13":
|
"encoding@npm:^0.1.13":
|
||||||
version: 0.1.13
|
version: 0.1.13
|
||||||
resolution: "encoding@npm:0.1.13"
|
resolution: "encoding@npm:0.1.13"
|
||||||
|
@ -4266,6 +4362,13 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"get-caller-file@npm:^2.0.1":
|
||||||
|
version: 2.0.5
|
||||||
|
resolution: "get-caller-file@npm:2.0.5"
|
||||||
|
checksum: b9769a836d2a98c3ee734a88ba712e62703f1df31b94b784762c433c27a386dd6029ff55c2a920c392e33657d80191edbf18c61487e198844844516f843496b9
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"get-intrinsic@npm:^1.0.2, get-intrinsic@npm:^1.1.0, get-intrinsic@npm:^1.1.1, get-intrinsic@npm:^1.1.3":
|
"get-intrinsic@npm:^1.0.2, get-intrinsic@npm:^1.1.0, get-intrinsic@npm:^1.1.1, get-intrinsic@npm:^1.1.3":
|
||||||
version: 1.1.3
|
version: 1.1.3
|
||||||
resolution: "get-intrinsic@npm:1.1.3"
|
resolution: "get-intrinsic@npm:1.1.3"
|
||||||
|
@ -6346,6 +6449,17 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"otplib@npm:^12.0.1":
|
||||||
|
version: 12.0.1
|
||||||
|
resolution: "otplib@npm:12.0.1"
|
||||||
|
dependencies:
|
||||||
|
"@otplib/core": ^12.0.1
|
||||||
|
"@otplib/preset-default": ^12.0.1
|
||||||
|
"@otplib/preset-v11": ^12.0.1
|
||||||
|
checksum: 4a1b91cf1b8e920b50ad4bac2ef2a89126630c62daf68e9b32ff15106b2551db905d3b979955cf5f8f114da0a8883cec3d636901d65e793c1745bb4174e2a572
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"p-filter@npm:2.1.0":
|
"p-filter@npm:2.1.0":
|
||||||
version: 2.1.0
|
version: 2.1.0
|
||||||
resolution: "p-filter@npm:2.1.0"
|
resolution: "p-filter@npm:2.1.0"
|
||||||
|
@ -6684,6 +6798,13 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"pngjs@npm:^5.0.0":
|
||||||
|
version: 5.0.0
|
||||||
|
resolution: "pngjs@npm:5.0.0"
|
||||||
|
checksum: 04e912cc45fb9601564e2284efaf0c5d20d131d9b596244f8a6789fc6cdb6b18d2975a6bbf7a001858d7e159d5c5c5dd7b11592e97629b7137f7f5cef05904c8
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"postcss@npm:8.4.14":
|
"postcss@npm:8.4.14":
|
||||||
version: 8.4.14
|
version: 8.4.14
|
||||||
resolution: "postcss@npm:8.4.14"
|
resolution: "postcss@npm:8.4.14"
|
||||||
|
@ -6876,6 +6997,20 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"qrcode@npm:^1.5.1":
|
||||||
|
version: 1.5.1
|
||||||
|
resolution: "qrcode@npm:1.5.1"
|
||||||
|
dependencies:
|
||||||
|
dijkstrajs: ^1.0.1
|
||||||
|
encode-utf8: ^1.0.3
|
||||||
|
pngjs: ^5.0.0
|
||||||
|
yargs: ^15.3.1
|
||||||
|
bin:
|
||||||
|
qrcode: bin/qrcode
|
||||||
|
checksum: 842f899d95caaad2ac507408b5498be3197e1df16bc6b537b20069d2cb1330e4588b50f672ce4a9ccf01338f7c97b5732ff9b5caaa6eb2338187d3c25a973e79
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"query-string@npm:^7.1.1":
|
"query-string@npm:^7.1.1":
|
||||||
version: 7.1.1
|
version: 7.1.1
|
||||||
resolution: "query-string@npm:7.1.1"
|
resolution: "query-string@npm:7.1.1"
|
||||||
|
@ -7154,6 +7289,20 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"require-directory@npm:^2.1.1":
|
||||||
|
version: 2.1.1
|
||||||
|
resolution: "require-directory@npm:2.1.1"
|
||||||
|
checksum: fb47e70bf0001fdeabdc0429d431863e9475e7e43ea5f94ad86503d918423c1543361cc5166d713eaa7029dd7a3d34775af04764bebff99ef413111a5af18c80
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"require-main-filename@npm:^2.0.0":
|
||||||
|
version: 2.0.0
|
||||||
|
resolution: "require-main-filename@npm:2.0.0"
|
||||||
|
checksum: e9e294695fea08b076457e9ddff854e81bffbe248ed34c1eec348b7abbd22a0d02e8d75506559e2265e96978f3c4720bd77a6dad84755de8162b357eb6c778c7
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"resolve-from@npm:^4.0.0":
|
"resolve-from@npm:^4.0.0":
|
||||||
version: 4.0.0
|
version: 4.0.0
|
||||||
resolution: "resolve-from@npm:4.0.0"
|
resolution: "resolve-from@npm:4.0.0"
|
||||||
|
@ -8043,6 +8192,13 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"thirty-two@npm:^1.0.2":
|
||||||
|
version: 1.0.2
|
||||||
|
resolution: "thirty-two@npm:1.0.2"
|
||||||
|
checksum: f6700b31d16ef942fdc0d14daed8a2f69ea8b60b0e85db8b83adf58d84bbeafe95a17d343ab55efaae571bb5148b62fc0ee12b04781323bf7af7d7e9693eec76
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"through2@npm:^3.0.1":
|
"through2@npm:^3.0.1":
|
||||||
version: 3.0.2
|
version: 3.0.2
|
||||||
resolution: "through2@npm:3.0.2"
|
resolution: "through2@npm:3.0.2"
|
||||||
|
@ -8468,6 +8624,13 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"which-module@npm:^2.0.0":
|
||||||
|
version: 2.0.0
|
||||||
|
resolution: "which-module@npm:2.0.0"
|
||||||
|
checksum: 809f7fd3dfcb2cdbe0180b60d68100c88785084f8f9492b0998c051d7a8efe56784492609d3f09ac161635b78ea29219eb1418a98c15ce87d085bce905705c9c
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"which-typed-array@npm:^1.1.2":
|
"which-typed-array@npm:^1.1.2":
|
||||||
version: 1.1.8
|
version: 1.1.8
|
||||||
resolution: "which-typed-array@npm:1.1.8"
|
resolution: "which-typed-array@npm:1.1.8"
|
||||||
|
@ -8578,6 +8741,13 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"y18n@npm:^4.0.0":
|
||||||
|
version: 4.0.3
|
||||||
|
resolution: "y18n@npm:4.0.3"
|
||||||
|
checksum: 014dfcd9b5f4105c3bb397c1c8c6429a9df004aa560964fb36732bfb999bfe83d45ae40aeda5b55d21b1ee53d8291580a32a756a443e064317953f08025b1aa4
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"yallist@npm:^4.0.0":
|
"yallist@npm:^4.0.0":
|
||||||
version: 4.0.0
|
version: 4.0.0
|
||||||
resolution: "yallist@npm:4.0.0"
|
resolution: "yallist@npm:4.0.0"
|
||||||
|
@ -8592,6 +8762,35 @@ __metadata:
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"yargs-parser@npm:^18.1.2":
|
||||||
|
version: 18.1.3
|
||||||
|
resolution: "yargs-parser@npm:18.1.3"
|
||||||
|
dependencies:
|
||||||
|
camelcase: ^5.0.0
|
||||||
|
decamelize: ^1.2.0
|
||||||
|
checksum: 60e8c7d1b85814594d3719300ecad4e6ae3796748b0926137bfec1f3042581b8646d67e83c6fc80a692ef08b8390f21ddcacb9464476c39bbdf52e34961dd4d9
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"yargs@npm:^15.3.1":
|
||||||
|
version: 15.4.1
|
||||||
|
resolution: "yargs@npm:15.4.1"
|
||||||
|
dependencies:
|
||||||
|
cliui: ^6.0.0
|
||||||
|
decamelize: ^1.2.0
|
||||||
|
find-up: ^4.1.0
|
||||||
|
get-caller-file: ^2.0.1
|
||||||
|
require-directory: ^2.1.1
|
||||||
|
require-main-filename: ^2.0.0
|
||||||
|
set-blocking: ^2.0.0
|
||||||
|
string-width: ^4.2.0
|
||||||
|
which-module: ^2.0.0
|
||||||
|
y18n: ^4.0.0
|
||||||
|
yargs-parser: ^18.1.2
|
||||||
|
checksum: 40b974f508d8aed28598087720e086ecd32a5fd3e945e95ea4457da04ee9bdb8bdd17fd91acff36dc5b7f0595a735929c514c40c402416bbb87c03f6fb782373
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"yocto-queue@npm:^0.1.0":
|
"yocto-queue@npm:^0.1.0":
|
||||||
version: 0.1.0
|
version: 0.1.0
|
||||||
resolution: "yocto-queue@npm:0.1.0"
|
resolution: "yocto-queue@npm:0.1.0"
|
||||||
|
@ -8635,6 +8834,7 @@ __metadata:
|
||||||
"@types/minio": ^7.0.14
|
"@types/minio": ^7.0.14
|
||||||
"@types/multer": ^1.4.7
|
"@types/multer": ^1.4.7
|
||||||
"@types/node": ^18.11.7
|
"@types/node": ^18.11.7
|
||||||
|
"@types/qrcode": ^1.5.0
|
||||||
"@types/react": ^18.0.24
|
"@types/react": ^18.0.24
|
||||||
"@types/sharp": ^0.31.0
|
"@types/sharp": ^0.31.0
|
||||||
argon2: ^0.30.1
|
argon2: ^0.30.1
|
||||||
|
@ -8659,8 +8859,10 @@ __metadata:
|
||||||
multer: ^1.4.5-lts.1
|
multer: ^1.4.5-lts.1
|
||||||
next: ^13.0.0
|
next: ^13.0.0
|
||||||
npm-run-all: ^4.1.5
|
npm-run-all: ^4.1.5
|
||||||
|
otplib: ^12.0.1
|
||||||
prettier: ^2.7.1
|
prettier: ^2.7.1
|
||||||
prisma: ^4.5.0
|
prisma: ^4.5.0
|
||||||
|
qrcode: ^1.5.1
|
||||||
react: ^18.2.0
|
react: ^18.2.0
|
||||||
react-chartjs-2: ^4.3.1
|
react-chartjs-2: ^4.3.1
|
||||||
react-dom: ^18.2.0
|
react-dom: ^18.2.0
|
||||||
|
|
Loading…
Add table
Reference in a new issue