2022-10-05 17:02:21 -05:00
|
|
|
import { HandlerContext, Handlers, PageProps } from "$fresh/server.ts";
|
|
|
|
import { compare } from "https://deno.land/x/bcrypt@v0.4.1/mod.ts";
|
|
|
|
import { Page } from "@/components/Page.tsx";
|
2022-10-07 23:22:35 -05:00
|
|
|
import { queryObject } from "@/db/mod.ts";
|
2022-09-30 15:14:57 -05:00
|
|
|
|
2022-10-05 17:02:21 -05:00
|
|
|
type UserID = string;
|
|
|
|
|
|
|
|
interface LoginError {
|
|
|
|
message: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function invalidLogin<LoginError>(context: HandlerContext<LoginError>) {
|
|
|
|
return await context.render({ message: "Invalid login" } as LoginError);
|
|
|
|
}
|
|
|
|
|
|
|
|
export const handler: Handlers<UserID | LoginError | null> = {
|
|
|
|
async POST(request: Request, context) {
|
|
|
|
const formData = (await request.formData());
|
|
|
|
const username = formData.get("username");
|
|
|
|
const password = formData.get("password");
|
|
|
|
if (!username) {
|
|
|
|
return await context.render({ message: "no username provided" });
|
|
|
|
}
|
|
|
|
if (!password) {
|
|
|
|
return await context.render({ message: "no password provided" });
|
|
|
|
}
|
|
|
|
|
2022-10-07 23:22:35 -05:00
|
|
|
const result = await queryObject<
|
2022-10-07 17:09:13 -05:00
|
|
|
{ id: string; username: string; password_digest: string }
|
2022-10-05 17:02:21 -05:00
|
|
|
>(
|
2022-10-07 23:22:35 -05:00
|
|
|
`select id, username, password_digest from "user" where username = $1`,
|
2022-10-05 17:02:21 -05:00
|
|
|
[username],
|
|
|
|
);
|
|
|
|
if (result == null || result.rows.length < 1) {
|
|
|
|
return await invalidLogin(context);
|
|
|
|
}
|
2022-10-07 17:09:13 -05:00
|
|
|
const { rows: [{ id, password_digest }] } = result;
|
|
|
|
if (await compare(password.toString(), password_digest)) {
|
2022-10-05 17:02:21 -05:00
|
|
|
return await context.render(id);
|
|
|
|
} else {
|
|
|
|
return await invalidLogin(context);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
export default function Login({ data }: PageProps) {
|
|
|
|
if (typeof data == "string") {
|
|
|
|
return LoginSuccessful(data);
|
|
|
|
} else {
|
|
|
|
return LoginForm(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function LoginSuccessful(_userId: UserID) {
|
|
|
|
return (
|
|
|
|
<Page>
|
|
|
|
<p>
|
|
|
|
You are now logged in. Let's go to your{" "}
|
|
|
|
<a href="/dashboard">dashboard</a>!
|
|
|
|
</p>
|
|
|
|
</Page>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function LoginForm(props?: LoginError | null) {
|
2022-09-30 15:14:57 -05:00
|
|
|
return (
|
|
|
|
<Page>
|
2022-10-05 17:02:21 -05:00
|
|
|
{props != null &&
|
|
|
|
(
|
|
|
|
<p class="text-red-500">
|
|
|
|
<strong>Error</strong>: {props.message}
|
|
|
|
</p>
|
|
|
|
)}
|
2022-10-01 14:34:07 -05:00
|
|
|
<h1 class="text-4xl mb-4 outline-white">
|
|
|
|
Log in to your account
|
|
|
|
</h1>
|
2022-09-30 15:14:57 -05:00
|
|
|
<form class="flex flex-col max-w-lg" method="post">
|
|
|
|
<label for="username">Username</label>
|
|
|
|
<input type="text" name="username" />
|
|
|
|
<label for="password">Password</label>
|
|
|
|
<input type="password" name="password" />
|
|
|
|
<input class="mt-2" type="submit" value="Login" />
|
|
|
|
</form>
|
|
|
|
</Page>
|
|
|
|
);
|
|
|
|
}
|