implement password change
This commit is contained in:
parent
9ffbdae1b6
commit
55d9e84431
|
@ -4,7 +4,7 @@ import './globals.css';
|
||||||
import AuthWrapper from '@/lib/components/AuthWrapper';
|
import AuthWrapper from '@/lib/components/AuthWrapper';
|
||||||
import { getServerSession } from 'next-auth';
|
import { getServerSession } from 'next-auth';
|
||||||
import NavigationPanel from '@/lib/components/ui/NavigationPanel';
|
import NavigationPanel from '@/lib/components/ui/NavigationPanel';
|
||||||
import { Flex } from '@radix-ui/themes';
|
import { Flex, ThemePanel } from '@radix-ui/themes';
|
||||||
|
|
||||||
export const metadata: Metadata = {
|
export const metadata: Metadata = {
|
||||||
title: 'Create Next App',
|
title: 'Create Next App',
|
||||||
|
|
|
@ -1,9 +1,10 @@
|
||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { Card, Text, Heading, Flex, TextField, Button, IconButton, Popover, Link, Tabs, Box, Grid } from "@radix-ui/themes";
|
import { changeOwnPassword } from "@/lib/actions";
|
||||||
import { CheckIcon, CopyIcon, InfoIcon, LockIcon, UserIcon } from "lucide-react";
|
import { Card, Text, Heading, Flex, TextField, Button, IconButton, Popover, Link, Tabs, Box, Grid, Dialog, Callout, Tooltip } from "@radix-ui/themes";
|
||||||
|
import { AlertCircleIcon, CheckIcon, CopyIcon, ExternalLinkIcon, InfoIcon, LockIcon, UserIcon } from "lucide-react";
|
||||||
import { useSession } from "next-auth/react";
|
import { useSession } from "next-auth/react";
|
||||||
import { useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
// TODO read these from environment
|
// TODO read these from environment
|
||||||
const EMAIL_HOST = "mx1.amogus.cloud";
|
const EMAIL_HOST = "mx1.amogus.cloud";
|
||||||
|
@ -11,10 +12,15 @@ const SMTP_PORT = "465";
|
||||||
const IMAP_PORT = "993";
|
const IMAP_PORT = "993";
|
||||||
const SMTP_SECURITY = "SSL/TLS";
|
const SMTP_SECURITY = "SSL/TLS";
|
||||||
const IMAP_SECURITY = "SSL/TLS";
|
const IMAP_SECURITY = "SSL/TLS";
|
||||||
|
const WEBMAIL_URL = "https://mail.amogus.cloud";
|
||||||
|
|
||||||
export default function SelfService() {
|
export default function SelfService() {
|
||||||
const session = useSession().data;
|
const session = useSession().data;
|
||||||
|
|
||||||
|
const [newPassword, setNewPassword] = useState("");
|
||||||
|
const validPassword = newPassword.length >= 8;
|
||||||
|
const [passwordChanged, setPasswordChanged] = useState(false);
|
||||||
|
|
||||||
function StaticValueField({ label, value }: { label: string, value: string }) {
|
function StaticValueField({ label, value }: { label: string, value: string }) {
|
||||||
const [checked, setChecked] = useState(false);
|
const [checked, setChecked] = useState(false);
|
||||||
|
|
||||||
|
@ -44,8 +50,31 @@ export default function SelfService() {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function PasswordChangedAlert() {
|
||||||
|
const [hide, setHide] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
let timeout = setTimeout(() => {
|
||||||
|
setHide(true);
|
||||||
|
setTimeout(() => setPasswordChanged(false), 500);
|
||||||
|
}, 5000);
|
||||||
|
return () => clearTimeout(timeout);
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Callout.Root className={`transition-all ${hide ? "opacity-0 h-0 m-0 !p-0" : " mb-4"}`}>
|
||||||
|
<Callout.Icon><CheckIcon /></Callout.Icon>
|
||||||
|
<Callout.Text>Password updated!</Callout.Text>
|
||||||
|
</Callout.Root>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
{passwordChanged && (
|
||||||
|
<PasswordChangedAlert />
|
||||||
|
)}
|
||||||
|
|
||||||
<Heading className="pb-4">Account settings</Heading>
|
<Heading className="pb-4">Account settings</Heading>
|
||||||
|
|
||||||
<Grid display="inline-grid" rows="1" columns="2" gap="4">
|
<Grid display="inline-grid" rows="1" columns="2" gap="4">
|
||||||
|
@ -83,9 +112,69 @@ export default function SelfService() {
|
||||||
<TextField.Input disabled value={"\u25CF".repeat(10)} />
|
<TextField.Input disabled value={"\u25CF".repeat(10)} />
|
||||||
</TextField.Root>
|
</TextField.Root>
|
||||||
|
|
||||||
<Button variant="outline">
|
<Dialog.Root>
|
||||||
Change
|
<Dialog.Trigger>
|
||||||
</Button>
|
<Button variant="outline">
|
||||||
|
Change
|
||||||
|
</Button>
|
||||||
|
</Dialog.Trigger>
|
||||||
|
|
||||||
|
<Dialog.Content>
|
||||||
|
<Dialog.Title>Update password</Dialog.Title>
|
||||||
|
|
||||||
|
<Box pb="4">
|
||||||
|
<Callout.Root>
|
||||||
|
<Callout.Icon>
|
||||||
|
<AlertCircleIcon />
|
||||||
|
</Callout.Icon>
|
||||||
|
<Callout.Text>You will be logged out of every configured mail client!</Callout.Text>
|
||||||
|
</Callout.Root>
|
||||||
|
</Box>
|
||||||
|
|
||||||
|
<TextField.Root>
|
||||||
|
<TextField.Slot>
|
||||||
|
<LockIcon height="16" width="16" />
|
||||||
|
</TextField.Slot>
|
||||||
|
<TextField.Input
|
||||||
|
type="password"
|
||||||
|
placeholder="New password"
|
||||||
|
value={newPassword}
|
||||||
|
onChange={(e) => setNewPassword(e.currentTarget.value)}
|
||||||
|
/>
|
||||||
|
</TextField.Root>
|
||||||
|
|
||||||
|
<Flex gap="3" mt="4" justify="end">
|
||||||
|
<Dialog.Close>
|
||||||
|
<Button variant="outline" onClick={() => setNewPassword("")}>Cancel</Button>
|
||||||
|
</Dialog.Close>
|
||||||
|
<Dialog.Close>
|
||||||
|
{validPassword
|
||||||
|
? (
|
||||||
|
<Button
|
||||||
|
variant="solid"
|
||||||
|
onClick={async () => {
|
||||||
|
setNewPassword("");
|
||||||
|
|
||||||
|
try {
|
||||||
|
setPasswordChanged(false);
|
||||||
|
await changeOwnPassword(newPassword);
|
||||||
|
setPasswordChanged(true);
|
||||||
|
} catch(e) {
|
||||||
|
alert(e);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>Update</Button>
|
||||||
|
)
|
||||||
|
: (
|
||||||
|
<Tooltip content="The password must have a minimum length of 8 characters">
|
||||||
|
<Button variant="solid" disabled>Update</Button>
|
||||||
|
</Tooltip>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</Dialog.Close>
|
||||||
|
</Flex>
|
||||||
|
</Dialog.Content>
|
||||||
|
</Dialog.Root>
|
||||||
</Flex>
|
</Flex>
|
||||||
</Flex>
|
</Flex>
|
||||||
</Card>
|
</Card>
|
||||||
|
@ -99,6 +188,11 @@ export default function SelfService() {
|
||||||
<Tabs.List size="2">
|
<Tabs.List size="2">
|
||||||
<Tabs.Trigger value="smtp">Outgoing</Tabs.Trigger>
|
<Tabs.Trigger value="smtp">Outgoing</Tabs.Trigger>
|
||||||
<Tabs.Trigger value="imap">Incoming</Tabs.Trigger>
|
<Tabs.Trigger value="imap">Incoming</Tabs.Trigger>
|
||||||
|
<Flex align="center" justify="end" grow="1" pr="3">
|
||||||
|
<Link href={WEBMAIL_URL} target="_blank">
|
||||||
|
<Button variant="ghost">Webmail <ExternalLinkIcon size="14" /></Button>
|
||||||
|
</Link>
|
||||||
|
</Flex>
|
||||||
</Tabs.List>
|
</Tabs.List>
|
||||||
|
|
||||||
<Box pt="3" pb="2">
|
<Box pt="3" pb="2">
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
"use server";
|
"use server";
|
||||||
|
|
||||||
import { database } from "./db";
|
import { getServerSession } from "next-auth";
|
||||||
|
import { database, setUserPassword } from "./db";
|
||||||
import { isAdmin } from "./util";
|
import { isAdmin } from "./util";
|
||||||
|
|
||||||
export async function fetchAllUsers(): Promise<string[]> {
|
export async function fetchAllUsers(): Promise<string[]> {
|
||||||
|
@ -15,3 +16,10 @@ export async function fetchAllUsers(): Promise<string[]> {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function changeOwnPassword(newPass: string) {
|
||||||
|
if (newPass.length < 8) throw new Error("Invalid password");
|
||||||
|
const session = await getServerSession();
|
||||||
|
if (!session?.user?.email) throw new Error("Unauthenticated");
|
||||||
|
await setUserPassword(session.user.email, newPass);
|
||||||
|
}
|
||||||
|
|
|
@ -32,3 +32,19 @@ export function validateCredentials(email: string, password: string) {
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function setUserPassword(email: string, newPass: string) {
|
||||||
|
return new Promise<void>(async (resolve, reject) => {
|
||||||
|
const db = database();
|
||||||
|
const hash = 'bcrypt:' + await bcrypt.hash(newPass, 10);
|
||||||
|
|
||||||
|
db.run("UPDATE passwords SET value = $1 WHERE key = $2",
|
||||||
|
{ 1: hash, 2: email },
|
||||||
|
async (err: any, res: any) => {
|
||||||
|
db.close();
|
||||||
|
if (err) return reject(err);
|
||||||
|
console.log(res);
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue