Skip to content

Commit 7117c6a

Browse files
committed
Add forget/reset password
close #50 Signed-off-by: gearnode <[email protected]>
1 parent ef8e62d commit 7117c6a

File tree

8 files changed

+632
-0
lines changed

8 files changed

+632
-0
lines changed

apps/console/src/App.tsx

+22
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,8 @@ const CreateFrameworkPage = lazy(() => import("./pages/CreateFrameworkPage"));
5252
const CreateControlPage = lazy(() => import("./pages/CreateControlPage"));
5353
const UpdateFrameworkPage = lazy(() => import("./pages/UpdateFrameworkPage"));
5454
const UpdateControlPage = lazy(() => import("./pages/UpdateControlPage"));
55+
const ForgotPasswordPage = lazy(() => import("./pages/ForgotPasswordPage"));
56+
const ResetPasswordPage = lazy(() => import("./pages/ResetPasswordPage"));
5557
// Policy pages
5658
const PolicyListPage = lazy(() => import("./pages/PolicyListPage"));
5759
const PolicyOverviewPage = lazy(() => import("./pages/PolicyOverviewPage"));
@@ -296,6 +298,26 @@ function App() {
296298
</Suspense>
297299
}
298300
/>
301+
<Route
302+
path="forgot-password"
303+
element={
304+
<Suspense>
305+
<VisitorErrorBoundaryWithLocation>
306+
<ForgotPasswordPage />
307+
</VisitorErrorBoundaryWithLocation>
308+
</Suspense>
309+
}
310+
/>
311+
<Route
312+
path="reset-password"
313+
element={
314+
<Suspense>
315+
<VisitorErrorBoundaryWithLocation>
316+
<ResetPasswordPage />
317+
</VisitorErrorBoundaryWithLocation>
318+
</Suspense>
319+
}
320+
/>
299321
<Route
300322
path="confirm-email"
301323
element={
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,125 @@
1+
import { useState } from "react";
2+
import { Link } from "react-router";
3+
import { Helmet } from "react-helmet-async";
4+
import { Button } from "@/components/ui/button";
5+
import { Input } from "@/components/ui/input";
6+
import { Label } from "@/components/ui/label";
7+
import { buildEndpoint } from "@/utils";
8+
import { useToast } from "@/hooks/use-toast";
9+
import {
10+
Card,
11+
CardContent,
12+
CardDescription,
13+
CardFooter,
14+
CardHeader,
15+
CardTitle,
16+
} from "@/components/ui/card";
17+
18+
export default function ForgotPasswordPage() {
19+
const [email, setEmail] = useState("");
20+
const [isLoading, setIsLoading] = useState(false);
21+
const [isSubmitted, setIsSubmitted] = useState(false);
22+
const { toast } = useToast();
23+
24+
const handleSubmit = async (e: React.FormEvent) => {
25+
e.preventDefault();
26+
setIsLoading(true);
27+
28+
try {
29+
const response = await fetch(
30+
buildEndpoint("/api/console/v1/auth/forget-password"),
31+
{
32+
method: "POST",
33+
headers: {
34+
"Content-Type": "application/json",
35+
},
36+
body: JSON.stringify({ email }),
37+
}
38+
);
39+
40+
if (!response.ok) {
41+
const error = await response.json();
42+
throw new Error(error.message || "Failed to request password reset");
43+
}
44+
45+
setIsSubmitted(true);
46+
toast({
47+
title: "Check your email",
48+
description: "If an account exists, you'll receive reset instructions",
49+
});
50+
} catch (error) {
51+
toast({
52+
title: "Error",
53+
description:
54+
error instanceof Error ? error.message : "An error occurred",
55+
variant: "destructive",
56+
});
57+
} finally {
58+
setIsLoading(false);
59+
}
60+
};
61+
62+
return (
63+
<>
64+
<Helmet>
65+
<title>Forgot Password - Probo</title>
66+
</Helmet>
67+
68+
<div className="flex flex-col items-center justify-center min-h-[70vh] p-4">
69+
<Card className="w-full max-w-md">
70+
<CardHeader>
71+
<CardTitle className="text-2xl font-bold text-center">
72+
Forgot Password
73+
</CardTitle>
74+
<CardDescription className="text-center">
75+
Enter your email address and we{"'"}ll send you a link to reset
76+
your password
77+
</CardDescription>
78+
</CardHeader>
79+
80+
<CardContent>
81+
{isSubmitted ? (
82+
<div className="space-y-4">
83+
<div className="p-3 text-sm text-green-600 bg-green-50 dark:bg-green-900/20 dark:text-green-400 rounded-md">
84+
Check your email for instructions to reset your password. If
85+
you don{"'"}t see it, check your spam folder.
86+
</div>
87+
<Button asChild className="w-full">
88+
<Link to="/login">Return to Login</Link>
89+
</Button>
90+
</div>
91+
) : (
92+
<form onSubmit={handleSubmit} className="space-y-4">
93+
<div className="space-y-2">
94+
<Label htmlFor="email">Email</Label>
95+
<Input
96+
id="email"
97+
type="email"
98+
placeholder="[email protected]"
99+
value={email}
100+
onChange={(e) => setEmail(e.target.value)}
101+
disabled={isLoading}
102+
required
103+
/>
104+
</div>
105+
106+
<Button type="submit" className="w-full" disabled={isLoading}>
107+
{isLoading ? "Sending..." : "Send Reset Link"}
108+
</Button>
109+
</form>
110+
)}
111+
</CardContent>
112+
113+
<CardFooter className="flex justify-center">
114+
<Link
115+
to="/login"
116+
className="text-sm text-gray-500 hover:text-gray-700 dark:text-gray-400 dark:hover:text-gray-300"
117+
>
118+
Back to Login
119+
</Link>
120+
</CardFooter>
121+
</Card>
122+
</div>
123+
</>
124+
);
125+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,205 @@
1+
import { useState, useEffect } from "react";
2+
import { useLocation, useNavigate } from "react-router";
3+
import { Helmet } from "react-helmet-async";
4+
import { Button } from "@/components/ui/button";
5+
import { Input } from "@/components/ui/input";
6+
import { Label } from "@/components/ui/label";
7+
import { buildEndpoint } from "@/utils";
8+
import { useToast } from "@/hooks/use-toast";
9+
import { Link } from "react-router";
10+
import {
11+
Card,
12+
CardContent,
13+
CardDescription,
14+
CardFooter,
15+
CardHeader,
16+
CardTitle,
17+
} from "@/components/ui/card";
18+
19+
export default function ResetPasswordPage() {
20+
const [isLoading, setIsLoading] = useState(false);
21+
const [isReset, setIsReset] = useState(false);
22+
const [error, setError] = useState<string | null>(null);
23+
const [token, setToken] = useState<string>("");
24+
const [password, setPassword] = useState<string>("");
25+
const [confirmPassword, setConfirmPassword] = useState<string>("");
26+
const location = useLocation();
27+
const navigate = useNavigate();
28+
const { toast } = useToast();
29+
30+
useEffect(() => {
31+
// Extract token from URL and prefill the form
32+
const searchParams = new URLSearchParams(location.search);
33+
const urlToken = searchParams.get("token");
34+
35+
if (urlToken) {
36+
setToken(urlToken);
37+
}
38+
}, [location.search]);
39+
40+
const handleSubmit = async (e: React.FormEvent) => {
41+
e.preventDefault();
42+
setIsLoading(true);
43+
setError(null);
44+
45+
if (!token.trim()) {
46+
setError("Please enter a reset token");
47+
setIsLoading(false);
48+
return;
49+
}
50+
51+
if (!password) {
52+
setError("Please enter a password");
53+
setIsLoading(false);
54+
return;
55+
}
56+
57+
if (password !== confirmPassword) {
58+
setError("Passwords do not match");
59+
setIsLoading(false);
60+
return;
61+
}
62+
63+
if (password.length < 8) {
64+
setError("Password must be at least 8 characters long");
65+
setIsLoading(false);
66+
return;
67+
}
68+
69+
try {
70+
const response = await fetch(
71+
buildEndpoint("/api/console/v1/auth/reset-password"),
72+
{
73+
method: "POST",
74+
headers: {
75+
"Content-Type": "application/json",
76+
},
77+
body: JSON.stringify({
78+
token: token.trim(),
79+
password: password,
80+
}),
81+
}
82+
);
83+
84+
const data = await response.json();
85+
86+
if (!response.ok) {
87+
throw new Error(data.message || "Failed to reset password");
88+
}
89+
90+
setIsReset(true);
91+
toast({
92+
title: "Success",
93+
description: "Your password has been reset successfully",
94+
});
95+
} catch (error) {
96+
setError(
97+
error instanceof Error
98+
? error.message
99+
: "Failed to reset password. Please try again."
100+
);
101+
} finally {
102+
setIsLoading(false);
103+
}
104+
};
105+
106+
return (
107+
<>
108+
<Helmet>
109+
<title>Reset Password - Probo</title>
110+
</Helmet>
111+
112+
<div className="flex flex-col items-center justify-center min-h-[70vh] p-4">
113+
<Card className="w-full max-w-md">
114+
<CardHeader>
115+
<CardTitle className="text-2xl font-bold text-center">
116+
Reset Password
117+
</CardTitle>
118+
<CardDescription className="text-center">
119+
Enter your new password below
120+
</CardDescription>
121+
</CardHeader>
122+
123+
<CardContent>
124+
{isReset ? (
125+
<div className="space-y-4 text-center">
126+
<p className="text-green-600 dark:text-green-400">
127+
Your password has been reset successfully!
128+
</p>
129+
<Button onClick={() => navigate("/login")} className="w-full">
130+
Proceed to Login
131+
</Button>
132+
</div>
133+
) : (
134+
<form onSubmit={handleSubmit} className="space-y-4">
135+
{error && (
136+
<div className="p-3 text-sm text-red-600 bg-red-50 dark:bg-red-900/20 dark:text-red-400 rounded-md">
137+
{error}
138+
</div>
139+
)}
140+
141+
<div className="space-y-2">
142+
<Label htmlFor="token">Reset Token</Label>
143+
<Input
144+
id="token"
145+
type="text"
146+
value={token}
147+
onChange={(e) => setToken(e.target.value)}
148+
placeholder="Enter your reset token"
149+
disabled={isLoading}
150+
required
151+
/>
152+
<p className="text-xs text-gray-500 dark:text-gray-400">
153+
The token has been automatically filled from the URL if
154+
available
155+
</p>
156+
</div>
157+
158+
<div className="space-y-2">
159+
<Label htmlFor="password">New Password</Label>
160+
<Input
161+
id="password"
162+
type="password"
163+
value={password}
164+
onChange={(e) => setPassword(e.target.value)}
165+
placeholder="Enter new password"
166+
disabled={isLoading}
167+
required
168+
/>
169+
</div>
170+
171+
<div className="space-y-2">
172+
<Label htmlFor="confirmPassword">Confirm New Password</Label>
173+
<Input
174+
id="confirmPassword"
175+
type="password"
176+
value={confirmPassword}
177+
onChange={(e) => setConfirmPassword(e.target.value)}
178+
placeholder="Confirm your new password"
179+
disabled={isLoading}
180+
required
181+
/>
182+
</div>
183+
184+
<Button type="submit" className="w-full" disabled={isLoading}>
185+
{isLoading ? "Resetting..." : "Reset Password"}
186+
</Button>
187+
</form>
188+
)}
189+
</CardContent>
190+
191+
<CardFooter className="flex justify-center">
192+
{!isReset && (
193+
<Link
194+
to="/login"
195+
className="text-sm text-gray-500 hover:text-gray-700 dark:text-gray-400 dark:hover:text-gray-300"
196+
>
197+
Back to Login
198+
</Link>
199+
)}
200+
</CardFooter>
201+
</Card>
202+
</div>
203+
</>
204+
);
205+
}

0 commit comments

Comments
 (0)