From 96cedbc09442b16793e60c1811b0c47cb24183c3 Mon Sep 17 00:00:00 2001 From: ritwik-69 <72665321+ritwik-69@users.noreply.github.com> Date: Tue, 4 Jun 2024 18:37:25 +0530 Subject: [PATCH] server:Implement JWT Authentication for User login and registration This commit adds the JWT authentication, user model using mongoose and login and register endpoints Fixes:#62 --- backend/middlewares/authMiddleware.js | 41 ++++++++++++ backend/models/userModel.ts | 21 +++++++ backend/package-lock.json | 90 ++++++++++++++++++++++++++- backend/package.json | 1 + backend/routes/userRoute.js | 10 +++ backend/src/index.ts | 6 ++ 6 files changed, 168 insertions(+), 1 deletion(-) create mode 100644 backend/middlewares/authMiddleware.js create mode 100644 backend/models/userModel.ts create mode 100644 backend/routes/userRoute.js diff --git a/backend/middlewares/authMiddleware.js b/backend/middlewares/authMiddleware.js new file mode 100644 index 0000000..62bb6dd --- /dev/null +++ b/backend/middlewares/authMiddleware.js @@ -0,0 +1,41 @@ +import jwt from 'jsonwebtoken'; +import User from '../models/userModel.js'; + +const authRoute = async (req, res, next) => { + try { + const token = req.cookies.jwt; + + if (!token) { + return res + .status(401) + .json({ error: 'Unauthorized - No Token Provided' }); + } + + const decoded = jwt.verify(token, process.env.JWT_SECRET); + + if (!decoded) { + return res + .status(401) + .json({ error: 'Unauthorized - Invalid Token' }); + } + + const user = await User.findById(decoded.userId).select('-password'); + + if (!user) { + return res.status(404).json({ error: 'User not found' }); + } + + req.user = user; + + next(); + } catch (error) { + console.log( + 'Error in middleware: ', + //@ts-expect-error error?.message is giving error due to ts configuration + error?.message || 'Invalid access token' + ); + res.status(500).json({ error: 'Internal server error' }); + } +}; + +export default authRoute; diff --git a/backend/models/userModel.ts b/backend/models/userModel.ts new file mode 100644 index 0000000..7304be5 --- /dev/null +++ b/backend/models/userModel.ts @@ -0,0 +1,21 @@ +import mongoose from 'mongoose'; + +const userSchema = new mongoose.Schema( + { + username: { + type: String, + required: true, + unique: true, + }, + password: { + type: String, + required: true, + minlength: 6, + }, + }, + { timestamps: true } +); + +const User = mongoose.model('User', userSchema); + +export default User; diff --git a/backend/package-lock.json b/backend/package-lock.json index 8a7aa85..da65df4 100644 --- a/backend/package-lock.json +++ b/backend/package-lock.json @@ -14,6 +14,7 @@ "cors": "^2.8.5", "dotenv": "^16.4.5", "express": "^4.19.2", + "jsonwebtoken": "^9.0.2", "mongoose": "^8.4.0", "ts-node": "^10.9.2", "uuid": "^9.0.1" @@ -2266,6 +2267,11 @@ "node": ">=16.20.1" } }, + "node_modules/buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==" + }, "node_modules/buffer-from": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", @@ -2800,6 +2806,14 @@ "url": "https://dotenvx.com" } }, + "node_modules/ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "dependencies": { + "safe-buffer": "^5.0.1" + } + }, "node_modules/ee-first": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", @@ -5270,6 +5284,46 @@ "node": ">=6" } }, + "node_modules/jsonwebtoken": { + "version": "9.0.2", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz", + "integrity": "sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==", + "dependencies": { + "jws": "^3.2.2", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=12", + "npm": ">=6" + } + }, + "node_modules/jwa": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", + "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", + "dependencies": { + "buffer-equal-constant-time": "1.0.1", + "ecdsa-sig-formatter": "1.0.11", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/jws": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", + "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", + "dependencies": { + "jwa": "^1.4.1", + "safe-buffer": "^5.0.1" + } + }, "node_modules/kareem": { "version": "2.6.3", "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.6.3.tgz", @@ -5339,6 +5393,36 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/lodash.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==" + }, + "node_modules/lodash.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==" + }, + "node_modules/lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==" + }, + "node_modules/lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==" + }, + "node_modules/lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==" + }, + "node_modules/lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==" + }, "node_modules/lodash.memoize": { "version": "4.1.2", "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", @@ -5351,6 +5435,11 @@ "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", "dev": true }, + "node_modules/lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==" + }, "node_modules/lru-cache": { "version": "5.1.1", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", @@ -6390,7 +6479,6 @@ "version": "7.6.2", "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.2.tgz", "integrity": "sha512-FNAIBWCx9qcRhoHcgcJ0gvU7SN1lYU2ZXuSfl04bSC5OpvDHFyJCjdNHomPXxjQlCBU67YW64PzY7/VIEH7F2w==", - "dev": true, "bin": { "semver": "bin/semver.js" }, diff --git a/backend/package.json b/backend/package.json index b23ce1c..8ff50a5 100644 --- a/backend/package.json +++ b/backend/package.json @@ -19,6 +19,7 @@ "cors": "^2.8.5", "dotenv": "^16.4.5", "express": "^4.19.2", + "jsonwebtoken": "^9.0.2", "mongoose": "^8.4.0", "ts-node": "^10.9.2", "uuid": "^9.0.1" diff --git a/backend/routes/userRoute.js b/backend/routes/userRoute.js new file mode 100644 index 0000000..311d224 --- /dev/null +++ b/backend/routes/userRoute.js @@ -0,0 +1,10 @@ +import express from 'express'; +import authRoute from '../middlewares/authMiddleware'; + +const router = express.Router(); + +router.post('/register', authRoute); + +router.post('/login', authRoute); + +export default router; diff --git a/backend/src/index.ts b/backend/src/index.ts index a6f0d71..1faa3a1 100644 --- a/backend/src/index.ts +++ b/backend/src/index.ts @@ -20,6 +20,12 @@ app.get('/', (req, res) => { res.send('Hello from the backend!'); }); +//routes + +import userRoute from '../routes/userRoute.js'; + +app.use('/api/v1/auth', userRoute); + app.listen(port, () => { console.log(`Server running on port ${port}`); });