ref: change look of oauth

This commit is contained in:
diced 2022-10-13 21:47:56 -07:00
parent 1a5925d7e8
commit a095768eae
No known key found for this signature in database
GPG key ID: 370BD1BA142842D1
7 changed files with 55 additions and 23 deletions

View file

@ -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,
}; };

View file

@ -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 {

View file

@ -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;
} }

View file

@ -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',

View file

@ -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',

View file

@ -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>

View file

@ -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>