From 13f97d40c2f11860b795f3386452ac294fba5fec Mon Sep 17 00:00:00 2001 From: Sourabh782 Date: Thu, 31 Oct 2024 17:29:59 +0530 Subject: [PATCH 1/2] implemented admin login/signup --- backend/controller/admin.controller.js | 7 +- backend/controller/customer.controller.js | 1 + frontend/src/components/Pages/Admin.jsx | 4 +- .../src/components/Pages/Admin/AdminLogin.jsx | 150 +++++++++++++++ .../components/Pages/Admin/AdminSignup.jsx | 182 ++++++++++++++++++ frontend/src/components/Pages/Login.jsx | 12 +- frontend/src/components/Pages/Signup.jsx | 2 +- .../Pages/{VerifyOtp.tsx => VerifyOtp.jsx} | 2 +- frontend/src/components/Shared/Navbar.jsx | 1 + frontend/src/context/userContext.jsx | 20 ++ frontend/src/main.jsx | 5 +- frontend/src/router/ProtectedRoute.jsx | 11 ++ frontend/src/router/index.jsx | 14 +- 13 files changed, 403 insertions(+), 8 deletions(-) create mode 100644 frontend/src/components/Pages/Admin/AdminLogin.jsx create mode 100644 frontend/src/components/Pages/Admin/AdminSignup.jsx rename frontend/src/components/Pages/{VerifyOtp.tsx => VerifyOtp.jsx} (97%) create mode 100644 frontend/src/context/userContext.jsx create mode 100644 frontend/src/router/ProtectedRoute.jsx diff --git a/backend/controller/admin.controller.js b/backend/controller/admin.controller.js index dbd74083..539cce26 100644 --- a/backend/controller/admin.controller.js +++ b/backend/controller/admin.controller.js @@ -73,11 +73,16 @@ async function loginAdmin(req, res) { const token = jwt.sign(payload, process.env.JWT_SECRET, { expiresIn: "1h", }); + res.cookie("authToken", token, { + maxAge: 1000 * 60 * 60, + httpOnly: true, + secure: true, + }); res.json({ message: "Login successful", token, role: "admin", - admin: { id: admin._id, name: admin.name, email: admin.email }, + admin: { id: admin._id, name: admin.name, email: admin.email, role: "admin" }, }); } catch (error) { logger.error("Error logging in admin:", { diff --git a/backend/controller/customer.controller.js b/backend/controller/customer.controller.js index b6195ece..be4ab012 100644 --- a/backend/controller/customer.controller.js +++ b/backend/controller/customer.controller.js @@ -143,6 +143,7 @@ async function loginCustomer(req, res) { id: customer._id, name: customer.name, email: customer.email, + role: "customer" }, }); } catch (error) { diff --git a/frontend/src/components/Pages/Admin.jsx b/frontend/src/components/Pages/Admin.jsx index bc588822..b2a48694 100644 --- a/frontend/src/components/Pages/Admin.jsx +++ b/frontend/src/components/Pages/Admin.jsx @@ -1,11 +1,13 @@ import { message } from 'antd'; import React, { useEffect, useState } from 'react'; import { useNavigate } from 'react-router-dom'; +import { useUser } from '../../context/userContext'; const Admin = () => { const [events, setEvents] = useState([]); const [error, setError] = useState(null); const navigate = useNavigate(); + const {user} = useUser(); // Fetch all events const fetchData = async () => { @@ -122,7 +124,7 @@ const Admin = () => {

- Hi {Admin.name}! + Hi {user.name}!

Welcome to Admin Panel diff --git a/frontend/src/components/Pages/Admin/AdminLogin.jsx b/frontend/src/components/Pages/Admin/AdminLogin.jsx new file mode 100644 index 00000000..8ecae30a --- /dev/null +++ b/frontend/src/components/Pages/Admin/AdminLogin.jsx @@ -0,0 +1,150 @@ +import React, { useState, useEffect } from 'react'; +import photo from '../../../assets/login.png'; +import { Link, useNavigate } from 'react-router-dom'; +import { message } from 'antd'; +import Cookies from 'js-cookie'; +import { FaEye } from 'react-icons/fa'; +import { FaEyeSlash } from 'react-icons/fa6'; +import { useUser } from '../../../context/userContext'; + +const AdminLogin = () => { + const API_URL = import.meta.env.VITE_BACKEND_URL || 'http://localhost:3000'; + const [data, setData] = useState({ + email: '', + password: '', + }); + const [hidden, setHidden] = useState(true); + const [isLoading, setIsLoading] = useState(false); + const [error, setError] = useState(null); + const {user, setUser} = useUser(); + + const navigate = useNavigate(); + + const handleChange = (e) => { + setData({ ...data, [e.target.name]: e.target.value }); + }; + + const handleSubmit = async (e) => { + e.preventDefault(); + setIsLoading(true); + setError(null); + try { + const response = await fetch(`${API_URL}/api/admin/login`, { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + }, + body: JSON.stringify(data), + }); + const result = await response.json(); + if (!response.ok) { + throw new Error(result.message || 'Login failed'); + } + const res = JSON.stringify(result.admin) + Cookies.set('authToken', result.token, { expires: 1, secure: true }); + Cookies.set("authenticatedUser", res, {expires: 1, secure: true}) + setUser(result.admin) + message.success('Login successful'); + navigate('/admin'); + } catch (err) { + setError(err.message || 'An error occurred. Please try again.'); + } finally { + setIsLoading(false); + } + }; + + useEffect(() => { + window.scrollTo(0, 0); + }, []); + + return ( +
+ {/* Background Image */} + login + {/* Login Form */} +
+
+ Admin Login, + + Log in to continue + +
+ + + +
+ + +
+ + + Forgot Password? + + +

+ Don’t have an account? + + Register Here + +

+ + + + + + {error &&

{error}

} + + +
+
+ ); +}; + +export default AdminLogin; diff --git a/frontend/src/components/Pages/Admin/AdminSignup.jsx b/frontend/src/components/Pages/Admin/AdminSignup.jsx new file mode 100644 index 00000000..903cfdf5 --- /dev/null +++ b/frontend/src/components/Pages/Admin/AdminSignup.jsx @@ -0,0 +1,182 @@ +import { useState, useEffect } from 'react'; +import photo from '../../../assets/login.png'; +import { useNavigate, Link } from 'react-router-dom'; +import { FaEye } from 'react-icons/fa'; +import { FaEyeSlash } from 'react-icons/fa6'; +import zxcvbn from 'zxcvbn'; // Password strength checker + +const AdminSignup = () => { + const API_URL = import.meta.env.VITE_BACKEND_URL || 'http://localhost:3000'; + const navigate = useNavigate(); + const [isLoading, setIsLoading] = useState(false); + const [error, setError] = useState(null); + const [passwordStrength, setPasswordStrength] = useState(0); + const [data, setData] = useState({ name: '', email: '', password: '' }); + const [hidden, setHidden] = useState(true); + + const handleChange = (e) => { + setData({ ...data, [e.target.name]: e.target.value }); + if (e.target.name === 'password') { + const result = zxcvbn(e.target.value); + setPasswordStrength(result.score); + } + }; + + const handleSubmit = async (e) => { + e.preventDefault(); + setIsLoading(true); + + // Input validation + if (!data.email || !data.password || !data.name) { + setError('Please fill in all fields'); + setIsLoading(false); + return; + } + if (data.password.length < 8) { + setError('Password must be at least 8 characters long'); + setIsLoading(false); + return; + } + if (data.name.length < 3) { + setError('Name must be at least 3 characters long'); + setIsLoading(false); + return; + } + if (!data.email.includes('@')) { + setError('Please enter a valid email address'); + setIsLoading(false); + return; + } + + try { + const response = await fetch(`${API_URL}/api/admin/register`, { + method: 'POST', + headers: { 'Content-Type': 'application/json' }, + body: JSON.stringify(data), + }); + const result = await response.json(); + + if (!response.ok) { + setIsLoading(false); + setError(result.error); + return; + } + + + // alert('OTP sent to your email. Verify to complete registration.'); + navigate('/admin-login'); + + } catch (error) { + setError(error.message); + console.error('Error:', error); + } finally { + setIsLoading(false); // Ensure loading state is reset after request + } + }; + + + useEffect(() => { + window.scrollTo(0, 0); + }, []); + + const getPasswordStrengthColor = (score) => { + const colors = ['#ff4d4d', '#ff944d', '#ffd24d', '#d2ff4d', '#4dff88']; + return colors[score]; + }; + + const getPasswordStrengthText = (score) => { + const strengthLevels = ['Very Weak', 'Weak', 'Okay', 'Good', 'Strong']; + return strengthLevels[score]; + }; + + return ( +
+ login +
+
+ Admin SignUp +
+ + Register to continue + +
+ + +
+ + +
+
+
+

+ Strength: {getPasswordStrengthText(passwordStrength)} +

+
+ {error && ( +
+ {error} +
+ )} +

+ Already have an account? + + Login + +

+ + + + +
+
+ ); +}; + +export default AdminSignup; diff --git a/frontend/src/components/Pages/Login.jsx b/frontend/src/components/Pages/Login.jsx index e39743d4..b8074a01 100644 --- a/frontend/src/components/Pages/Login.jsx +++ b/frontend/src/components/Pages/Login.jsx @@ -35,13 +35,21 @@ const Login = () => { body: JSON.stringify(data), }); const result = await response.json(); - if (!response.ok) { + console.log(result); + + if (!response) { throw new Error(result.message || 'Login failed'); } - Cookies.set('authToken', result.token, { expire: '1h', secure: true }); + const res = JSON.stringify(result.user) + + Cookies.set('authToken', result.token, { expires: 1, secure: true }); + Cookies.set("authenticatedUser", res, {expires: 1, secure: true}) + message.success('Login successful'); navigate('/'); } catch (err) { + console.log(err); + setError(err.message || 'An error occurred. Please try again.'); } finally { setIsLoading(false); diff --git a/frontend/src/components/Pages/Signup.jsx b/frontend/src/components/Pages/Signup.jsx index 37498c37..ce678fe6 100644 --- a/frontend/src/components/Pages/Signup.jsx +++ b/frontend/src/components/Pages/Signup.jsx @@ -95,7 +95,7 @@ const Signup = () => { src={photo} alt="login" loading="lazy" - className="w-3/4 absolute inset-0" + className="absolute w-3/4 lg:w-auto lg:opacity-100 opacity-10 object-cover" />
diff --git a/frontend/src/components/Pages/VerifyOtp.tsx b/frontend/src/components/Pages/VerifyOtp.jsx similarity index 97% rename from frontend/src/components/Pages/VerifyOtp.tsx rename to frontend/src/components/Pages/VerifyOtp.jsx index 135bca55..2e5b1ccb 100644 --- a/frontend/src/components/Pages/VerifyOtp.tsx +++ b/frontend/src/components/Pages/VerifyOtp.jsx @@ -4,7 +4,7 @@ import React, { useState } from 'react'; import { message } from 'antd'; const VerifyOtp = () => { - const API_URL = import.meta.env.VITE_BACKEND_URL || 'http://localhost:3000'; + const API_URL = process.env.VITE_BACKEND_URL || 'http://localhost:3000'; const navigate = useNavigate(); // Use useNavigate for navigation const [otp, setOtp] = useState("") diff --git a/frontend/src/components/Shared/Navbar.jsx b/frontend/src/components/Shared/Navbar.jsx index e01dd381..e787c552 100644 --- a/frontend/src/components/Shared/Navbar.jsx +++ b/frontend/src/components/Shared/Navbar.jsx @@ -55,6 +55,7 @@ const Navbar = () => { }) Cookies.remove('authToken'); + Cookies.remove("authenticatedUser"); setToken(null); setIsModalOpen(false); // Close the modal setIsMenuOpen(false); // after getting logged out close the menu if it is open diff --git a/frontend/src/context/userContext.jsx b/frontend/src/context/userContext.jsx new file mode 100644 index 00000000..45c53c05 --- /dev/null +++ b/frontend/src/context/userContext.jsx @@ -0,0 +1,20 @@ +import React, { createContext, useState, useContext } from 'react'; +import Cookies from 'js-cookie'; + +const UserContext = createContext(); + +export const UserProvider = ({ children }) => { + const data = Cookies.get("authenticatedUser"); + + const userData = data ? JSON.parse(data) : null; + console.log(userData); + const [user, setUser] = useState(userData); + + return ( + + {children} + + ); +}; + +export const useUser = () => useContext(UserContext); diff --git a/frontend/src/main.jsx b/frontend/src/main.jsx index 0853bb3a..49c8e916 100644 --- a/frontend/src/main.jsx +++ b/frontend/src/main.jsx @@ -3,9 +3,12 @@ import ReactDOM from 'react-dom/client'; import './index.css'; import router from './router'; import { RouterProvider } from 'react-router-dom'; +import { UserProvider } from './context/userContext.jsx'; ReactDOM.createRoot(document.getElementById('root')).render( - + + + ); diff --git a/frontend/src/router/ProtectedRoute.jsx b/frontend/src/router/ProtectedRoute.jsx new file mode 100644 index 00000000..fc36a6f1 --- /dev/null +++ b/frontend/src/router/ProtectedRoute.jsx @@ -0,0 +1,11 @@ +import React from 'react'; +import { Route, Navigate } from 'react-router-dom'; +import { useUser } from '../context/userContext'; + +const ProtectedRoute = ({ children }) => { + const {user} = useUser(); + + return user.role == "admin" ? children : ; +}; + +export default ProtectedRoute; \ No newline at end of file diff --git a/frontend/src/router/index.jsx b/frontend/src/router/index.jsx index 83ca6a13..0e66022f 100644 --- a/frontend/src/router/index.jsx +++ b/frontend/src/router/index.jsx @@ -22,6 +22,9 @@ import VerifyOtp from '../components/Pages/VerifyOtp'; import EmailVerify from '../components/Pages/EmailVerify'; import Membership from '../components/Membership'; import OtpRegisterVerify from '../components/Pages/verifyRegisterOtp'; +import AdminLogin from '../components/Pages/Admin/AdminLogin'; +import AdminSignup from '../components/Pages/Admin/AdminSignup'; +import ProtectedRoute from './ProtectedRoute'; const router = createBrowserRouter( createRoutesFromElements( }> @@ -36,7 +39,16 @@ const router = createBrowserRouter( } /> } /> } /> - } /> + + + + } + /> + } /> + } /> } /> } /> } /> From 7c0982847eeaf3b6f230861dac42cad2c37a3da8 Mon Sep 17 00:00:00 2001 From: Sourabh782 Date: Sat, 2 Nov 2024 16:48:25 +0530 Subject: [PATCH 2/2] included suggestions --- backend/controller/admin.controller.js | 2 +- frontend/src/components/Pages/Admin.jsx | 2 +- .../src/components/Pages/Admin/AdminLogin.jsx | 15 +++++++++--- .../components/Pages/Admin/AdminSignup.jsx | 7 +++--- frontend/src/components/Pages/Login.jsx | 4 ++-- frontend/src/components/Pages/VerifyOtp.jsx | 2 +- frontend/src/context/userContext.jsx | 23 +++++++++++++++---- frontend/src/router/ProtectedRoute.jsx | 6 ++++- 8 files changed, 45 insertions(+), 16 deletions(-) diff --git a/backend/controller/admin.controller.js b/backend/controller/admin.controller.js index 539cce26..14997a77 100644 --- a/backend/controller/admin.controller.js +++ b/backend/controller/admin.controller.js @@ -82,7 +82,7 @@ async function loginAdmin(req, res) { message: "Login successful", token, role: "admin", - admin: { id: admin._id, name: admin.name, email: admin.email, role: "admin" }, + admin: { id: admin._id, name: admin.name, email: admin.email, role: admin.role || "admin" }, }); } catch (error) { logger.error("Error logging in admin:", { diff --git a/frontend/src/components/Pages/Admin.jsx b/frontend/src/components/Pages/Admin.jsx index b2a48694..99bf816d 100644 --- a/frontend/src/components/Pages/Admin.jsx +++ b/frontend/src/components/Pages/Admin.jsx @@ -267,7 +267,7 @@ const Admin = () => {
{error &&

{error}

} - {events.map((event) => ( + {events.length > 0 && events.map((event) => (
{ const handleSubmit = async (e) => { e.preventDefault(); + const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/; + if (!emailRegex.test(data.email)) { + setError('Please enter a valid email address'); + return; + } + if (data.password.length < 8) { + setError('Password must be at least 8 characters long'); + return; + } setIsLoading(true); setError(null); try { @@ -122,8 +131,8 @@ const AdminLogin = () => {

- - + {error &&

{error}

} diff --git a/frontend/src/components/Pages/Admin/AdminSignup.jsx b/frontend/src/components/Pages/Admin/AdminSignup.jsx index 903cfdf5..cc5baf9b 100644 --- a/frontend/src/components/Pages/Admin/AdminSignup.jsx +++ b/frontend/src/components/Pages/Admin/AdminSignup.jsx @@ -42,7 +42,8 @@ const AdminSignup = () => { setIsLoading(false); return; } - if (!data.email.includes('@')) { + const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/; + if (!emailRegex.test(data.email)) { setError('Please enter a valid email address'); setIsLoading(false); return; @@ -163,11 +164,11 @@ const AdminSignup = () => { Login - + - +