ref: change look of oauth
This commit is contained in:
parent
1a5925d7e8
commit
a095768eae
7 changed files with 55 additions and 23 deletions
|
@ -27,6 +27,8 @@ import ExternalLinkIcon from './ExternalLinkIcon';
|
||||||
import ShareXIcon from './ShareXIcon';
|
import ShareXIcon from './ShareXIcon';
|
||||||
import DownloadIcon from './DownloadIcon';
|
import DownloadIcon from './DownloadIcon';
|
||||||
import FlameshotIcon from './FlameshotIcon';
|
import FlameshotIcon from './FlameshotIcon';
|
||||||
|
import GitHubIcon from './GitHubIcon';
|
||||||
|
import DiscordIcon from './DiscordIcon';
|
||||||
|
|
||||||
export {
|
export {
|
||||||
ActivityIcon,
|
ActivityIcon,
|
||||||
|
@ -58,4 +60,6 @@ export {
|
||||||
ShareXIcon,
|
ShareXIcon,
|
||||||
DownloadIcon,
|
DownloadIcon,
|
||||||
FlameshotIcon,
|
FlameshotIcon,
|
||||||
|
GitHubIcon,
|
||||||
|
DiscordIcon,
|
||||||
};
|
};
|
|
@ -5,18 +5,18 @@ import { GetServerSideProps } from 'next';
|
||||||
|
|
||||||
export const getServerSideProps: GetServerSideProps = async ctx => {
|
export const getServerSideProps: GetServerSideProps = async ctx => {
|
||||||
// this entire thing will also probably change before the stable release
|
// this entire thing will also probably change before the stable release
|
||||||
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 oauth_providers = [];
|
const oauth_providers = [];
|
||||||
|
|
||||||
if (ghEnabled) oauth_providers.push({
|
if (ghEnabled) oauth_providers.push({
|
||||||
name: 'GitHub',
|
name: 'GitHub',
|
||||||
url: github_auth.oauth_url(config.oauth.github_client_id),
|
url: '/api/auth/oauth/github',
|
||||||
});
|
});
|
||||||
if (discEnabled) oauth_providers.push({
|
if (discEnabled) oauth_providers.push({
|
||||||
name: 'Discord',
|
name: 'Discord',
|
||||||
url: discord_auth.oauth_url(config.oauth.discord_client_id, `${config.core.https ? 'https' : 'http'}://${ctx.req.headers.host}`),
|
url: '/api/auth/oauth/discord',
|
||||||
});
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -26,7 +26,7 @@ export function createToken() {
|
||||||
}
|
}
|
||||||
|
|
||||||
export function sign(value: string, secret: string): string {
|
export function sign(value: string, secret: string): string {
|
||||||
const signed = value + ':' + createHmac('sha256', secret)
|
const signed = value + ':' + createHmac('sha256', secret)
|
||||||
.update(value)
|
.update(value)
|
||||||
.digest('base64')
|
.digest('base64')
|
||||||
.replace(/=+$/, '');
|
.replace(/=+$/, '');
|
||||||
|
@ -54,7 +54,7 @@ export function unsign64(value: string, secret: string): string {
|
||||||
}
|
}
|
||||||
|
|
||||||
export function chunk<T>(arr: T[], size: number): Array<T[]> {
|
export function chunk<T>(arr: T[], size: number): Array<T[]> {
|
||||||
const result = [];
|
const result = [];
|
||||||
const L = arr.length;
|
const L = arr.length;
|
||||||
let i = 0;
|
let i = 0;
|
||||||
|
|
||||||
|
@ -67,7 +67,7 @@ export function chunk<T>(arr: T[], size: number): Array<T[]> {
|
||||||
|
|
||||||
export async function sizeOfDir(directory: string): Promise<number> {
|
export async function sizeOfDir(directory: string): Promise<number> {
|
||||||
const files = await readdir(directory);
|
const files = await readdir(directory);
|
||||||
|
|
||||||
let size = 0;
|
let size = 0;
|
||||||
for (let i = 0, L = files.length; i !== L; ++i) {
|
for (let i = 0, L = files.length; i !== L; ++i) {
|
||||||
const sta = await stat(join(directory, files[i]));
|
const sta = await stat(join(directory, files[i]));
|
||||||
|
@ -92,7 +92,7 @@ export function bytesToRead(bytes: number) {
|
||||||
export function randomInvis(length: number) {
|
export function randomInvis(length: number) {
|
||||||
// some parts from https://github.com/tycrek/ass/blob/master/generators/lengthGen.js
|
// some parts from https://github.com/tycrek/ass/blob/master/generators/lengthGen.js
|
||||||
const invisibleCharset = ['\u200B', '\u2060', '\u200C', '\u200D'];
|
const invisibleCharset = ['\u200B', '\u2060', '\u200C', '\u200D'];
|
||||||
|
|
||||||
return [...randomBytes(length)].map((byte) => invisibleCharset[Number(byte) % invisibleCharset.length]).join('').slice(1).concat(invisibleCharset[0]);
|
return [...randomBytes(length)].map((byte) => invisibleCharset[Number(byte) % invisibleCharset.length]).join('').slice(1).concat(invisibleCharset[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -107,7 +107,7 @@ export function createInvisImage(length: number, imageId: number) {
|
||||||
});
|
});
|
||||||
|
|
||||||
if (existing) return retry();
|
if (existing) return retry();
|
||||||
|
|
||||||
const inv = await prisma.invisibleImage.create({
|
const inv = await prisma.invisibleImage.create({
|
||||||
data: {
|
data: {
|
||||||
invis,
|
invis,
|
||||||
|
@ -132,7 +132,7 @@ export function createInvisURL(length: number, urlId: string) {
|
||||||
});
|
});
|
||||||
|
|
||||||
if (existing) return retry();
|
if (existing) return retry();
|
||||||
|
|
||||||
const ur = await prisma.invisibleUrl.create({
|
const ur = await prisma.invisibleUrl.create({
|
||||||
data: {
|
data: {
|
||||||
invis,
|
invis,
|
||||||
|
@ -156,6 +156,6 @@ export async function getBase64URLFromURL(url: string) {
|
||||||
return `data:${res.headers.get('content-type')};base64,${base64}`;
|
return `data:${res.headers.get('content-type')};base64,${base64}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function notNull(a: any, b: any) {
|
export function notNull(a: any, b: any) {
|
||||||
return a !== null && b !== null;
|
return a !== null && b !== null;
|
||||||
}
|
}
|
|
@ -14,7 +14,7 @@ async function handler(req: NextApiReq, res: NextApiRes) {
|
||||||
}
|
}
|
||||||
|
|
||||||
const { code } = req.query as { code: string };
|
const { code } = req.query as { code: string };
|
||||||
if (!code) return res.bad('no code');
|
if (!code) return res.redirect(discord_auth.oauth_url(config.oauth.discord_client_id, `${config.core.https ? 'https' : 'http'}://${req.headers.host}`));
|
||||||
|
|
||||||
const resp = await fetch('https://discord.com/api/oauth2/token', {
|
const resp = await fetch('https://discord.com/api/oauth2/token', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
|
|
|
@ -15,7 +15,7 @@ async function handler(req: NextApiReq, res: NextApiRes) {
|
||||||
|
|
||||||
const { code } = req.query as { code: string };
|
const { code } = req.query as { code: string };
|
||||||
|
|
||||||
if (!code) return res.bad('no code');
|
if (!code) return res.redirect(github_auth.oauth_url(config.oauth.github_client_id));
|
||||||
|
|
||||||
const resp = await fetch('https://github.com/login/oauth/access_token', {
|
const resp = await fetch('https://github.com/login/oauth/access_token', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
|
|
|
@ -1,14 +1,27 @@
|
||||||
import { Button, Center, TextInput, Title, PasswordInput } from '@mantine/core';
|
import { Button, Center, TextInput, Title, PasswordInput, Divider } from '@mantine/core';
|
||||||
import { useForm } from '@mantine/form';
|
import { useForm } from '@mantine/form';
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import useFetch from 'hooks/useFetch';
|
import useFetch from 'hooks/useFetch';
|
||||||
import { useRouter } from 'next/router';
|
import { useRouter } from 'next/router';
|
||||||
import { useEffect } from 'react';
|
import { useEffect } from 'react';
|
||||||
|
import Head from 'next/head';
|
||||||
|
import { GitHubIcon, DiscordIcon } from 'components/icons';
|
||||||
export { getServerSideProps } from 'middleware/getServerSideProps';
|
export { getServerSideProps } from 'middleware/getServerSideProps';
|
||||||
|
|
||||||
export default function Login({ oauth_registration }) {
|
export default function Login({ title, oauth_registration, oauth_providers: unparsed }) {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
|
const oauth_providers = JSON.parse(unparsed);
|
||||||
|
|
||||||
|
const icons = {
|
||||||
|
GitHub: GitHubIcon,
|
||||||
|
Discord: DiscordIcon,
|
||||||
|
};
|
||||||
|
|
||||||
|
for (const provider of oauth_providers) {
|
||||||
|
provider.Icon = icons[provider.name];
|
||||||
|
}
|
||||||
|
|
||||||
const form = useForm({
|
const form = useForm({
|
||||||
initialValues: {
|
initialValues: {
|
||||||
username: '',
|
username: '',
|
||||||
|
@ -47,9 +60,12 @@ export default function Login({ oauth_registration }) {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
<Head>
|
||||||
|
<title>{title} - Login</title>
|
||||||
|
</Head>
|
||||||
<Center sx={{ height: '100vh' }}>
|
<Center sx={{ height: '100vh' }}>
|
||||||
<div>
|
<div>
|
||||||
<Title align='center'>Zipline</Title>
|
<Title align='center'>{title}</Title>
|
||||||
<form onSubmit={form.onSubmit((v) => onSubmit(v))}>
|
<form onSubmit={form.onSubmit((v) => onSubmit(v))}>
|
||||||
<TextInput size='lg' id='username' label='Username' {...form.getInputProps('username')} />
|
<TextInput size='lg' id='username' label='Username' {...form.getInputProps('username')} />
|
||||||
<PasswordInput size='lg' id='password' label='Password' {...form.getInputProps('password')} />
|
<PasswordInput size='lg' id='password' label='Password' {...form.getInputProps('password')} />
|
||||||
|
@ -57,9 +73,14 @@ export default function Login({ oauth_registration }) {
|
||||||
<Button size='lg' type='submit' fullWidth mt={12}>Login</Button>
|
<Button size='lg' type='submit' fullWidth mt={12}>Login</Button>
|
||||||
</form>
|
</form>
|
||||||
{oauth_registration && (
|
{oauth_registration && (
|
||||||
<Link href='/auth/register' passHref>
|
<>
|
||||||
<Button size='lg' fullWidth mt={12} component='a'>Register</Button>
|
<Divider label='or' labelPosition='center' my={8} />
|
||||||
</Link>
|
{oauth_providers.map(({ url, name, Icon }, i) => (
|
||||||
|
<Link key={i} href={url} passHref>
|
||||||
|
<Button size='lg' fullWidth leftIcon={<Icon />} component='a' my={8}>Login in with {name}</Button>
|
||||||
|
</Link>
|
||||||
|
))}
|
||||||
|
</>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</Center>
|
</Center>
|
||||||
|
|
|
@ -1,14 +1,13 @@
|
||||||
import { Button, Center, TextInput, Title, PasswordInput } from '@mantine/core';
|
import { Button, Center } from '@mantine/core';
|
||||||
import { useForm } from '@mantine/form';
|
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import useFetch from 'hooks/useFetch';
|
|
||||||
import { useRouter } from 'next/router';
|
import { useRouter } from 'next/router';
|
||||||
import { useEffect } from 'react';
|
import { useEffect } from 'react';
|
||||||
import GitHubIcon from 'components/icons/GitHubIcon';
|
import GitHubIcon from 'components/icons/GitHubIcon';
|
||||||
import DiscordIcon from 'components/icons/DiscordIcon';
|
import DiscordIcon from 'components/icons/DiscordIcon';
|
||||||
|
import Head from 'next/head';
|
||||||
export { getServerSideProps } from 'middleware/getServerSideProps';
|
export { getServerSideProps } from 'middleware/getServerSideProps';
|
||||||
|
|
||||||
export default function Login({ oauth_registration, oauth_providers: unparsed }) {
|
export default function Login({ title, oauth_registration, oauth_providers: unparsed }) {
|
||||||
const oauth_providers = JSON.parse(unparsed);
|
const oauth_providers = JSON.parse(unparsed);
|
||||||
|
|
||||||
const icons = {
|
const icons = {
|
||||||
|
@ -35,8 +34,16 @@ export default function Login({ oauth_registration, oauth_providers: unparsed })
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
<Head>
|
||||||
|
<title>{title} - Login</title>
|
||||||
|
</Head>
|
||||||
<Center sx={{ height: '100vh' }}>
|
<Center sx={{ height: '100vh' }}>
|
||||||
<div>
|
<div>
|
||||||
|
<Link href='/auth/login' passHref>
|
||||||
|
<Button size='lg' fullWidth variant='outline' component='a'>
|
||||||
|
Go Back to Login
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
{oauth_providers.map(({ url, name, Icon }, i) => (
|
{oauth_providers.map(({ url, name, Icon }, i) => (
|
||||||
<Link key={i} href={url} passHref>
|
<Link key={i} href={url} passHref>
|
||||||
<Button size='lg' fullWidth mt={12} leftIcon={<Icon />} component='a'>Sign in with {name}</Button>
|
<Button size='lg' fullWidth mt={12} leftIcon={<Icon />} component='a'>Sign in with {name}</Button>
|
||||||
|
|
Loading…
Reference in a new issue