Login route
This commit is contained in:
parent
1f9dd51200
commit
52bb27a6d4
2 changed files with 26 additions and 9 deletions
|
|
@ -1,13 +1,14 @@
|
||||||
import { NextRequest, NextResponse } from "next/server";
|
import { NextRequest, NextResponse } from "next/server";
|
||||||
|
|
||||||
export async function POST(req: NextRequest) {
|
export async function POST(req: NextRequest) {
|
||||||
const { username, password } = await req.json();
|
const { username, password, totp } = await req.json();
|
||||||
|
|
||||||
const res = await fetch("http://localhost:3001/auth/login", {
|
const res = await fetch("http://localhost:3001/auth/login", {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
Authorization:
|
Authorization:
|
||||||
"Basic " + Buffer.from(`${username}:${password}`).toString("base64"),
|
"Basic " +
|
||||||
|
Buffer.from(`${username}:${password}${totp}`).toString("base64"),
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,4 @@
|
||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { useRouter } from "next/navigation";
|
import { useRouter } from "next/navigation";
|
||||||
|
|
||||||
|
|
@ -7,6 +6,7 @@ export default function AuthPage() {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const [username, setUsername] = useState("");
|
const [username, setUsername] = useState("");
|
||||||
const [password, setPassword] = useState("");
|
const [password, setPassword] = useState("");
|
||||||
|
const [totp, setTotp] = useState("");
|
||||||
const [error, setError] = useState("");
|
const [error, setError] = useState("");
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
const [checking, setChecking] = useState(true);
|
const [checking, setChecking] = useState(true);
|
||||||
|
|
@ -32,20 +32,17 @@ export default function AuthPage() {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
setError("");
|
setError("");
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const res = await fetch("/api/auth/login", {
|
const res = await fetch("/api/auth/login", {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: { "Content-Type": "application/json" },
|
headers: { "Content-Type": "application/json" },
|
||||||
body: JSON.stringify({ username, password }),
|
body: JSON.stringify({ username, password, totp }),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!res.ok) {
|
if (!res.ok) {
|
||||||
setError("Invalid username or password.");
|
setError("Invalid credentials or authenticator code.");
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const callbackUrl =
|
const callbackUrl =
|
||||||
new URLSearchParams(window.location.search).get("callbackUrl") ?? "/";
|
new URLSearchParams(window.location.search).get("callbackUrl") ?? "/";
|
||||||
router.push(callbackUrl);
|
router.push(callbackUrl);
|
||||||
|
|
@ -88,7 +85,7 @@ export default function AuthPage() {
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Password */}
|
{/* Password */}
|
||||||
<div className="mb-6">
|
<div className="mb-4">
|
||||||
<label className="block text-[11px] tracking-wider text-gray-400 uppercase mb-1.5">
|
<label className="block text-[11px] tracking-wider text-gray-400 uppercase mb-1.5">
|
||||||
Password
|
Password
|
||||||
</label>
|
</label>
|
||||||
|
|
@ -102,6 +99,25 @@ export default function AuthPage() {
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* TOTP */}
|
||||||
|
<div className="mb-6">
|
||||||
|
<label className="block text-[11px] tracking-wider text-gray-400 uppercase mb-1.5">
|
||||||
|
Authenticator code
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={totp}
|
||||||
|
onChange={(e) =>
|
||||||
|
setTotp(e.target.value.replace(/\D/g, "").slice(0, 6))
|
||||||
|
}
|
||||||
|
required
|
||||||
|
autoComplete="one-time-code"
|
||||||
|
inputMode="numeric"
|
||||||
|
placeholder="000000"
|
||||||
|
className="w-full px-3.5 py-2.5 border border-gray-200 rounded-xl text-sm text-gray-900 bg-gray-50 outline-none focus:border-blue-300 focus:ring-2 focus:ring-blue-50 transition-colors tracking-widest"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Error */}
|
{/* Error */}
|
||||||
{error && <p className="text-[13px] text-red-400 mb-4">{error}</p>}
|
{error && <p className="text-[13px] text-red-400 mb-4">{error}</p>}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue