├── .gitignore ├── controllers ├── authController.js ├── categoryController.js ├── productController.js └── userController.js ├── index.js ├── middlewares ├── auth.js ├── user.js └── userValidator.js ├── models ├── category.js ├── product.js └── user.js ├── package-lock.json ├── package.json └── routes ├── auth.js ├── categories.js ├── products.js └── users.js /.gitignore: -------------------------------------------------------------------------------- 1 | /node_modules 2 | .env -------------------------------------------------------------------------------- /controllers/authController.js: -------------------------------------------------------------------------------- 1 | const User = require('../models/user'); 2 | const jwt = require('jsonwebtoken'); 3 | 4 | exports.salam = (req, res) => { 5 | res.send({ message: 'users module' }) 6 | } 7 | 8 | exports.signup = (req, res) => { 9 | 10 | const user = new User(req.body); 11 | 12 | user.save((err, user) => { 13 | if(err) { 14 | return res.status(400).send(err) 15 | } 16 | 17 | res.send(user) 18 | }) 19 | } 20 | 21 | exports.signin = (req, res) => { 22 | 23 | const { email, password } = req.body; 24 | 25 | User.findOne({email}, (err, user) => { 26 | 27 | if(err || !user) { 28 | return res.status(400).json({ 29 | error: 'User not found with this email, Please SignUp!' 30 | }) 31 | } 32 | 33 | if(!user.authenticate(password)) { 34 | return res.status(401).json({ 35 | error: 'Email and Password dont Match !' 36 | }) 37 | } 38 | 39 | const token = jwt.sign({_id: user._id, role: user.role}, process.env.JWT_SECRET); 40 | 41 | res.cookie('token', token, {expire: new Date() + 8062000}) 42 | 43 | const { _id, name, email, role } = user; 44 | 45 | return res.json({ 46 | token, user: {_id, name, email, role} 47 | }) 48 | 49 | }) 50 | 51 | } 52 | 53 | exports.signout = (req, res) => { 54 | 55 | res.clearCookie('token'); 56 | 57 | res.json({ 58 | message: "User Signout" 59 | }) 60 | 61 | } 62 | -------------------------------------------------------------------------------- /controllers/categoryController.js: -------------------------------------------------------------------------------- 1 | const Category = require('../models/category'); 2 | 3 | exports.createCategory = (req, res) => { 4 | 5 | const category = new Category(req.body); 6 | 7 | category.save((err, category) => { 8 | 9 | if(err) { 10 | return res.status(400).json({ 11 | error: 'bad Request !' 12 | }) 13 | } 14 | 15 | res.json({ 16 | cartegory: category 17 | }) 18 | }) 19 | 20 | } 21 | 22 | exports.categoryId = (req, res, next, id) => { 23 | 24 | Category.findById(id).exec((err, category) => { 25 | 26 | if(err || !category) { 27 | return res.status(404).json({ 28 | error: "Category not found !" 29 | }) 30 | } 31 | 32 | req.category = category; 33 | next() 34 | }) 35 | 36 | } 37 | 38 | 39 | exports.showCategory = (req, res) => { 40 | 41 | let category = req.category; 42 | 43 | res.json({ 44 | category 45 | }) 46 | } 47 | 48 | 49 | exports.updateCategory = (req, res) => { 50 | 51 | let category = req.category; 52 | 53 | category.name = req.body.name; 54 | 55 | category.save((err, category) => { 56 | 57 | if(err) { 58 | return res.status(400).json({ 59 | error: "bad request !" 60 | }) 61 | } 62 | 63 | res.json({ 64 | category, 65 | message: 'Category updated ' 66 | }) 67 | 68 | }) 69 | 70 | } 71 | 72 | 73 | exports.deleteCategory = (req, res) => { 74 | 75 | let category = req.category; 76 | 77 | category.remove((err, category) => { 78 | 79 | if(err) { 80 | return res.status(404).json({ 81 | error: "category not found !" 82 | }) 83 | } 84 | 85 | res.status(204).json({ 86 | message: 'Category deleted ' 87 | }) 88 | 89 | }) 90 | 91 | } 92 | 93 | exports.allCategories = (req, res) => { 94 | 95 | Category.find().exec((err, categories) => { 96 | if(err){ 97 | return res.status(500).json({ 98 | error: err 99 | }) 100 | } 101 | 102 | res.json({ 103 | categories 104 | }) 105 | }) 106 | } 107 | 108 | 109 | 110 | -------------------------------------------------------------------------------- /controllers/productController.js: -------------------------------------------------------------------------------- 1 | const Product = require('../models/product'); 2 | const _ = require('lodash'); 3 | const fs = require('fs'); 4 | const Joi = require('joi'); 5 | 6 | const formidable = require('formidable') 7 | 8 | exports.createProduct = (req, res) => { 9 | 10 | let form = new formidable.IncomingForm(); 11 | 12 | form.keepExtensions = true; 13 | 14 | form.parse(req, (err, fields, files) => { 15 | 16 | if(err) { 17 | return res.status(400).json({ 18 | error: 'Image could not uploaded !' 19 | }) 20 | } 21 | 22 | 23 | let product = new Product(fields); 24 | 25 | if(files.photo) { 26 | 27 | if(files.photo.size > Math.pow(10, 6)) { 28 | return res.status(400).json({ 29 | error: 'Image should be less than 1mb in size !' 30 | }) 31 | } 32 | 33 | product.photo.data = fs.readFileSync(files.photo.path) 34 | product.photo.contentType = files.photo.type 35 | } 36 | 37 | const schema = Joi.object({ 38 | name: Joi.string().required(), 39 | description: Joi.string().required(), 40 | price: Joi.required(), 41 | quantity: Joi.required(), 42 | category: Joi.required() 43 | }) 44 | 45 | const { error } = schema.validate(fields); 46 | 47 | if(error) { 48 | return res.status(400).json({ 49 | error: error.details[0].message 50 | }) 51 | } 52 | 53 | product.save((err, product) => { 54 | if(err) { 55 | return res.status(400).json({ 56 | err: 'Product not persist ' 57 | }) 58 | } 59 | 60 | res.json({ 61 | product 62 | }) 63 | }) 64 | 65 | }) 66 | } 67 | 68 | 69 | 70 | exports.productById = (req, res, next, id) => { 71 | 72 | Product.findById(id).exec((err, product) => { 73 | 74 | if(err || !product) { 75 | return res.status(404).json({ 76 | error: 'Product not found !' 77 | }) 78 | } 79 | 80 | req.product = product; 81 | next() 82 | 83 | }) 84 | 85 | } 86 | 87 | 88 | exports.showProduct = (req, res) => { 89 | 90 | req.product.photo = undefined; 91 | 92 | res.json({ 93 | product: req.product 94 | }) 95 | } 96 | 97 | 98 | exports.updateProduct = (req, res) => { 99 | 100 | let form = new formidable.IncomingForm(); 101 | 102 | form.keepExtensions = true; 103 | 104 | form.parse(req, (err, fields, files) => { 105 | 106 | if(err) { 107 | return res.status(400).json({ 108 | error: 'Image could not uploaded !' 109 | }) 110 | } 111 | 112 | 113 | let product = req.product; 114 | 115 | product = _.extend(product, fields) 116 | 117 | 118 | if(files.photo) { 119 | 120 | if(files.photo.size > Math.pow(10, 6)) { 121 | return res.status(400).json({ 122 | error: 'Image should be less than 1mb in size !' 123 | }) 124 | } 125 | 126 | product.photo.data = fs.readFileSync(files.photo.path) 127 | product.photo.contentType = files.photo.type 128 | } 129 | 130 | const schema = Joi.object({ 131 | name: Joi.string().required(), 132 | description: Joi.string().required(), 133 | price: Joi.required(), 134 | quantity: Joi.required(), 135 | category: Joi.required() 136 | }) 137 | 138 | const { error } = schema.validate(fields); 139 | 140 | if(error) { 141 | return res.status(400).json({ 142 | error: error.details[0].message 143 | }) 144 | } 145 | 146 | product.save((err, product) => { 147 | if(err) { 148 | return res.status(400).json({ 149 | err: 'Product not updated ' 150 | }) 151 | } 152 | 153 | res.json({ 154 | product 155 | }) 156 | }) 157 | 158 | }) 159 | } 160 | 161 | 162 | 163 | exports.productById = (req, res, next, id) => { 164 | 165 | Product.findById(id).exec((err, product) => { 166 | 167 | if(err || !product) { 168 | return res.status(404).json({ 169 | error: 'Product not found !' 170 | }) 171 | } 172 | 173 | req.product = product; 174 | next() 175 | 176 | }) 177 | 178 | } 179 | 180 | 181 | exports.showProduct = (req, res) => { 182 | 183 | req.product.photo = undefined; 184 | 185 | res.json({ 186 | product: req.product 187 | }) 188 | } 189 | 190 | 191 | exports.removeProduct = (req, res) => { 192 | 193 | let product = req.product 194 | 195 | product.remove((err, product) => { 196 | 197 | if(err) { 198 | return res.status(404).json({ 199 | error: "Product not found !" 200 | }) 201 | } 202 | 203 | res.status(204).json({}) 204 | 205 | }) 206 | 207 | } 208 | 209 | exports.allProducts = (req, res) => { 210 | 211 | let sortBy = req.query.sortBy ? req.query.sortBy : '_id'; 212 | let order = req.query.order ? req.query.order : 'asc'; 213 | let limit = req.query.limit ? parseInt(req.query.limit) : 100; 214 | 215 | Product.find() 216 | .select("-photo") 217 | .populate('category') 218 | .sort([[sortBy, order]]) 219 | .limit(limit) 220 | .exec((err, products) => { 221 | 222 | if(err) { 223 | return res.status(404).json({ 224 | error: "Products not found !" 225 | }) 226 | } 227 | 228 | res.json({ 229 | products 230 | }) 231 | }) 232 | 233 | } 234 | 235 | exports.relatedProduct = (req, res) => { 236 | 237 | let limit = req.query.limit ? parseInt(req.query.limit) : 6; 238 | 239 | Product.find({category: req.product.category, _id: { $ne: req.product._id }}) 240 | .limit(limit) 241 | .select('-photo') 242 | .populate('category', '_id name') 243 | .exec((err, products) => { 244 | 245 | if(err) { 246 | return res.status(404).json({ 247 | error: "Products not found !" 248 | }) 249 | } 250 | 251 | res.json({ 252 | products 253 | }) 254 | 255 | }) 256 | 257 | } 258 | 259 | exports.SearchProduct = (req, res) => { 260 | 261 | let sortBy = req.query.sortBy ? req.query.sortBy : '_id'; 262 | let order = req.query.order ? req.query.order : 'asc'; 263 | let limit = req.query.limit ? parseInt(req.query.limit) : 100; 264 | let skip = parseInt(req.body.skip); 265 | let findArgs = {}; 266 | 267 | 268 | for (let key in req.body.filters) { 269 | if (req.body.filters[key].length > 0) { 270 | if (key === "price") { 271 | // gte - greater than price [0-10] 272 | // lte - less than 273 | findArgs[key] = { 274 | $gte: req.body.filters[key][0], 275 | $lte: req.body.filters[key][1] 276 | }; 277 | } else { 278 | findArgs[key] = req.body.filters[key]; 279 | } 280 | } 281 | } 282 | 283 | Product.find(findArgs) 284 | .select("-photo") 285 | .populate('category') 286 | .sort([[sortBy, order]]) 287 | .limit(limit) 288 | .skip(skip) 289 | .exec((err, products) => { 290 | 291 | if(err) { 292 | return res.status(404).json({ 293 | error: "Products not found !" 294 | }) 295 | } 296 | 297 | res.json({ 298 | products 299 | }) 300 | }) 301 | 302 | } 303 | 304 | exports.photoProduct = (req, res) => { 305 | 306 | const { data, contentType } = req.product.photo; 307 | 308 | if(data) { 309 | 310 | res.set('Content-Type', contentType) 311 | 312 | return res.send(data) 313 | 314 | } 315 | 316 | } 317 | 318 | 319 | 320 | 321 | 322 | 323 | -------------------------------------------------------------------------------- /controllers/userController.js: -------------------------------------------------------------------------------- 1 | const User = require("../models/user") 2 | 3 | exports.getOneUser = (req, res) => { 4 | 5 | req.profile.hashed_password = undefined 6 | req.profile.salt = undefined 7 | 8 | res.json({ 9 | user: req.profile 10 | }) 11 | 12 | } 13 | 14 | exports.updateOneUser = (req, res) => { 15 | 16 | User.findOneAndUpdate({_id: req.profile._id}, {$set: req.body}, {new: true}, (err, user) => { 17 | 18 | if(err) { 19 | return res.status(400).json({err}) 20 | } 21 | 22 | user.hashed_password = undefined 23 | user.salt = undefined 24 | 25 | res.json({user}) 26 | 27 | }) 28 | 29 | } 30 | -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const cors = require('cors'); 3 | const mongoose = require('mongoose'); 4 | const cookieParser = require('cookie-parser'); 5 | const expressValidator = require('express-validator') 6 | 7 | //Import Routes 8 | const authRoutes = require('./routes/auth'); 9 | const userRoutes = require('./routes/users'); 10 | const categoryRoutes = require('./routes/categories'); 11 | const productRoutes = require('./routes/products'); 12 | 13 | //Config App 14 | require('dotenv').config(); 15 | const app = express(); 16 | 17 | //Db mongoDB 18 | mongoose.connect(process.env.DATABASE, { 19 | useNewUrlParser: true, 20 | useCreateIndex: true, 21 | useUnifiedTopology: true 22 | }) 23 | .then(() => console.log('db connected')) 24 | .catch(() => console.log('not nonnect to the database !')) 25 | 26 | //Middlewares 27 | app.use(express.json()) 28 | app.use(cors()) 29 | app.use(cookieParser()) 30 | app.use(expressValidator()) 31 | 32 | //Routes Middleware 33 | app.use('/api', authRoutes); 34 | app.use('/api/user', userRoutes); 35 | app.use('/api/category', categoryRoutes); 36 | app.use('/api/product', productRoutes); 37 | 38 | 39 | const port = process.env.PORT || 3000; 40 | app.listen(port, () => console.log(`app is running on port ${port}`)); -------------------------------------------------------------------------------- /middlewares/auth.js: -------------------------------------------------------------------------------- 1 | const expressJWT = require('express-jwt'); 2 | require('dotenv').config(); 3 | 4 | exports.requireSignIn = expressJWT({ 5 | secret: process.env.JWT_SECRET, 6 | algorithms: ["HS256"], 7 | userProperty: 'auth' 8 | }) 9 | 10 | 11 | exports.isAuth = (req, res, next) => { 12 | 13 | let user = req.profile && req.auth && (req.profile._id == req.auth._id) 14 | 15 | if(!user) { 16 | return res.status(403).json({ 17 | error: "Access Denied" 18 | }) 19 | } 20 | 21 | next() 22 | } 23 | 24 | 25 | exports.isAdmin = (req, res, next) => { 26 | 27 | if(req.auth.role == 0) { 28 | return res.status(403).json({ 29 | error: "Admin Resource, Access Denied !" 30 | }) 31 | } 32 | 33 | next(); 34 | 35 | } -------------------------------------------------------------------------------- /middlewares/user.js: -------------------------------------------------------------------------------- 1 | const User = require('../models/user'); 2 | 3 | exports.userById = (req, res, next, id) => { 4 | 5 | User.findById(id).exec((err, user) => { 6 | 7 | if(err || !user) { 8 | return res.status(404).json({ 9 | error: "user not found !" 10 | }) 11 | } 12 | 13 | req.profile = user; 14 | next(); 15 | }) 16 | 17 | } -------------------------------------------------------------------------------- /middlewares/userValidator.js: -------------------------------------------------------------------------------- 1 | exports.userSignUpValidator = (req, res, next) => { 2 | 3 | req.check('name', 'Name is Required !').notEmpty(); 4 | 5 | req.check('email', 'Email is Required !') 6 | .notEmpty() 7 | .isEmail(); 8 | 9 | req.check('password', 'Password is Required !') 10 | .notEmpty() 11 | .isLength({min: 6, max: 10}) 12 | .withMessage('Password must between 6 and 10 Caracters') 13 | 14 | const errors = req.validationErrors() 15 | 16 | if(errors) 17 | { 18 | return res.status(400).json(errors) 19 | } 20 | 21 | next() 22 | } -------------------------------------------------------------------------------- /models/category.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose'); 2 | 3 | const categorySchema = new mongoose.Schema({ 4 | name: { 5 | type: String, 6 | require: true, 7 | maxLength: 32, 8 | trim: true 9 | } 10 | }, {timestamps: true}); 11 | 12 | module.exports = mongoose.model('Category', categorySchema); -------------------------------------------------------------------------------- /models/product.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose'); 2 | const { ObjectId } = mongoose.Schema; 3 | 4 | const productSchema = new mongoose.Schema({ 5 | name: { 6 | type: String, 7 | require: true, 8 | maxLength: 150, 9 | trim: true 10 | }, 11 | description: { 12 | type: String, 13 | require: true, 14 | maxLength: 2000 15 | }, 16 | price: { 17 | type: Number, 18 | require: true 19 | }, 20 | quantity: { 21 | type: Number 22 | }, 23 | photo: { 24 | data: Buffer, 25 | contentType: String 26 | }, 27 | category: { 28 | type: ObjectId, 29 | ref: 'Category', 30 | require: true 31 | }, 32 | shipping: { 33 | type: Boolean, 34 | require: false, 35 | default: false 36 | } 37 | }, {timestamps: true}); 38 | 39 | module.exports = mongoose.model('Product', productSchema); -------------------------------------------------------------------------------- /models/user.js: -------------------------------------------------------------------------------- 1 | const mongoose = require('mongoose'); 2 | const crypto = require('crypto'); 3 | const { v1: uuid } = require('uuid'); 4 | 5 | const userSchema = new mongoose.Schema({ 6 | name: { 7 | type: String, 8 | trim: true, 9 | maxlength: 50, 10 | required: true 11 | }, 12 | email: { 13 | type: String, 14 | trim: true, 15 | maxlength: 50, 16 | required: true, 17 | unique: true 18 | }, 19 | hashed_password:{ 20 | type: String, 21 | required: true, 22 | }, 23 | salt: { 24 | type: String 25 | }, 26 | about:{ 27 | type: String, 28 | trim: true 29 | }, 30 | role: { 31 | type: Number, 32 | default: 0 33 | }, 34 | history: { 35 | type: Array, 36 | default: [] 37 | } 38 | }, {timestamps: true}) 39 | 40 | 41 | userSchema.virtual('password') 42 | .set(function(password){ 43 | this._password = password; 44 | this.salt = uuid(); 45 | this.hashed_password = this.cryptPassword(password) 46 | }) 47 | .get(function() { 48 | return this._password; 49 | }) 50 | 51 | userSchema.methods = { 52 | authenticate: function(plainText) { 53 | return this.cryptPassword(plainText) === this.hashed_password; 54 | }, 55 | cryptPassword: function(password) { 56 | if(!password) return ''; 57 | 58 | try { 59 | 60 | return crypto 61 | .createHmac('sha1', this.salt) 62 | .update(password) 63 | .digest('hex'); 64 | 65 | } catch (error) { 66 | return '' 67 | } 68 | } 69 | } 70 | 71 | module.exports = mongoose.model('User', userSchema); -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ecommerce", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@hapi/address": { 8 | "version": "4.1.0", 9 | "resolved": "https://registry.npmjs.org/@hapi/address/-/address-4.1.0.tgz", 10 | "integrity": "sha512-SkszZf13HVgGmChdHo/PxchnSaCJ6cetVqLzyciudzZRT0jcOouIF/Q93mgjw8cce+D+4F4C1Z/WrfFN+O3VHQ==", 11 | "requires": { 12 | "@hapi/hoek": "^9.0.0" 13 | } 14 | }, 15 | "@hapi/formula": { 16 | "version": "2.0.0", 17 | "resolved": "https://registry.npmjs.org/@hapi/formula/-/formula-2.0.0.tgz", 18 | "integrity": "sha512-V87P8fv7PI0LH7LiVi8Lkf3x+KCO7pQozXRssAHNXXL9L1K+uyu4XypLXwxqVDKgyQai6qj3/KteNlrqDx4W5A==" 19 | }, 20 | "@hapi/hoek": { 21 | "version": "9.1.0", 22 | "resolved": "https://registry.npmjs.org/@hapi/hoek/-/hoek-9.1.0.tgz", 23 | "integrity": "sha512-i9YbZPN3QgfighY/1X1Pu118VUz2Fmmhd6b2n0/O8YVgGGfw0FbUYoA97k7FkpGJ+pLCFEDLUmAPPV4D1kpeFw==" 24 | }, 25 | "@hapi/pinpoint": { 26 | "version": "2.0.0", 27 | "resolved": "https://registry.npmjs.org/@hapi/pinpoint/-/pinpoint-2.0.0.tgz", 28 | "integrity": "sha512-vzXR5MY7n4XeIvLpfl3HtE3coZYO4raKXW766R6DZw/6aLqR26iuZ109K7a0NtF2Db0jxqh7xz2AxkUwpUFybw==" 29 | }, 30 | "@hapi/topo": { 31 | "version": "5.0.0", 32 | "resolved": "https://registry.npmjs.org/@hapi/topo/-/topo-5.0.0.tgz", 33 | "integrity": "sha512-tFJlT47db0kMqVm3H4nQYgn6Pwg10GTZHb1pwmSiv1K4ks6drQOtfEF5ZnPjkvC+y4/bUPHK+bc87QvLcL+WMw==", 34 | "requires": { 35 | "@hapi/hoek": "^9.0.0" 36 | } 37 | }, 38 | "accepts": { 39 | "version": "1.3.7", 40 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", 41 | "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", 42 | "requires": { 43 | "mime-types": "~2.1.24", 44 | "negotiator": "0.6.2" 45 | } 46 | }, 47 | "array-flatten": { 48 | "version": "1.1.1", 49 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 50 | "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" 51 | }, 52 | "async": { 53 | "version": "1.5.2", 54 | "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz", 55 | "integrity": "sha1-7GphrlZIDAw8skHJVhjiCJL5Zyo=" 56 | }, 57 | "bl": { 58 | "version": "2.2.0", 59 | "resolved": "https://registry.npmjs.org/bl/-/bl-2.2.0.tgz", 60 | "integrity": "sha512-wbgvOpqopSr7uq6fJrLH8EsvYMJf9gzfo2jCsL2eTy75qXPukA4pCgHamOQkZtY5vmfVtjB+P3LNlMHW5CEZXA==", 61 | "requires": { 62 | "readable-stream": "^2.3.5", 63 | "safe-buffer": "^5.1.1" 64 | } 65 | }, 66 | "bluebird": { 67 | "version": "3.5.1", 68 | "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz", 69 | "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA==" 70 | }, 71 | "body-parser": { 72 | "version": "1.19.0", 73 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", 74 | "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", 75 | "requires": { 76 | "bytes": "3.1.0", 77 | "content-type": "~1.0.4", 78 | "debug": "2.6.9", 79 | "depd": "~1.1.2", 80 | "http-errors": "1.7.2", 81 | "iconv-lite": "0.4.24", 82 | "on-finished": "~2.3.0", 83 | "qs": "6.7.0", 84 | "raw-body": "2.4.0", 85 | "type-is": "~1.6.17" 86 | } 87 | }, 88 | "bson": { 89 | "version": "1.1.5", 90 | "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.5.tgz", 91 | "integrity": "sha512-kDuEzldR21lHciPQAIulLs1LZlCXdLziXI6Mb/TDkwXhb//UORJNPXgcRs2CuO4H0DcMkpfT3/ySsP3unoZjBg==" 92 | }, 93 | "buffer-equal-constant-time": { 94 | "version": "1.0.1", 95 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", 96 | "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" 97 | }, 98 | "bytes": { 99 | "version": "3.1.0", 100 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", 101 | "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" 102 | }, 103 | "content-disposition": { 104 | "version": "0.5.3", 105 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", 106 | "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", 107 | "requires": { 108 | "safe-buffer": "5.1.2" 109 | } 110 | }, 111 | "content-type": { 112 | "version": "1.0.4", 113 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", 114 | "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" 115 | }, 116 | "cookie": { 117 | "version": "0.4.0", 118 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", 119 | "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" 120 | }, 121 | "cookie-parser": { 122 | "version": "1.4.5", 123 | "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.5.tgz", 124 | "integrity": "sha512-f13bPUj/gG/5mDr+xLmSxxDsB9DQiTIfhJS/sqjrmfAWiAN+x2O4i/XguTL9yDZ+/IFDanJ+5x7hC4CXT9Tdzw==", 125 | "requires": { 126 | "cookie": "0.4.0", 127 | "cookie-signature": "1.0.6" 128 | } 129 | }, 130 | "cookie-signature": { 131 | "version": "1.0.6", 132 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 133 | "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" 134 | }, 135 | "core-util-is": { 136 | "version": "1.0.2", 137 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", 138 | "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" 139 | }, 140 | "cors": { 141 | "version": "2.8.5", 142 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 143 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 144 | "requires": { 145 | "object-assign": "^4", 146 | "vary": "^1" 147 | } 148 | }, 149 | "debug": { 150 | "version": "2.6.9", 151 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 152 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 153 | "requires": { 154 | "ms": "2.0.0" 155 | } 156 | }, 157 | "denque": { 158 | "version": "1.4.1", 159 | "resolved": "https://registry.npmjs.org/denque/-/denque-1.4.1.tgz", 160 | "integrity": "sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ==" 161 | }, 162 | "depd": { 163 | "version": "1.1.2", 164 | "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", 165 | "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" 166 | }, 167 | "destroy": { 168 | "version": "1.0.4", 169 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", 170 | "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" 171 | }, 172 | "dotenv": { 173 | "version": "8.2.0", 174 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz", 175 | "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw==" 176 | }, 177 | "ecdsa-sig-formatter": { 178 | "version": "1.0.11", 179 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", 180 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", 181 | "requires": { 182 | "safe-buffer": "^5.0.1" 183 | } 184 | }, 185 | "ee-first": { 186 | "version": "1.1.1", 187 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 188 | "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" 189 | }, 190 | "encodeurl": { 191 | "version": "1.0.2", 192 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 193 | "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" 194 | }, 195 | "escape-html": { 196 | "version": "1.0.3", 197 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 198 | "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" 199 | }, 200 | "etag": { 201 | "version": "1.8.1", 202 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 203 | "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" 204 | }, 205 | "express": { 206 | "version": "4.17.1", 207 | "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", 208 | "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", 209 | "requires": { 210 | "accepts": "~1.3.7", 211 | "array-flatten": "1.1.1", 212 | "body-parser": "1.19.0", 213 | "content-disposition": "0.5.3", 214 | "content-type": "~1.0.4", 215 | "cookie": "0.4.0", 216 | "cookie-signature": "1.0.6", 217 | "debug": "2.6.9", 218 | "depd": "~1.1.2", 219 | "encodeurl": "~1.0.2", 220 | "escape-html": "~1.0.3", 221 | "etag": "~1.8.1", 222 | "finalhandler": "~1.1.2", 223 | "fresh": "0.5.2", 224 | "merge-descriptors": "1.0.1", 225 | "methods": "~1.1.2", 226 | "on-finished": "~2.3.0", 227 | "parseurl": "~1.3.3", 228 | "path-to-regexp": "0.1.7", 229 | "proxy-addr": "~2.0.5", 230 | "qs": "6.7.0", 231 | "range-parser": "~1.2.1", 232 | "safe-buffer": "5.1.2", 233 | "send": "0.17.1", 234 | "serve-static": "1.14.1", 235 | "setprototypeof": "1.1.1", 236 | "statuses": "~1.5.0", 237 | "type-is": "~1.6.18", 238 | "utils-merge": "1.0.1", 239 | "vary": "~1.1.2" 240 | } 241 | }, 242 | "express-jwt": { 243 | "version": "6.0.0", 244 | "resolved": "https://registry.npmjs.org/express-jwt/-/express-jwt-6.0.0.tgz", 245 | "integrity": "sha512-C26y9myRjx7CyhZ+BAT3p+gQyRCoDZ7qo8plCvLDaRT6je6ALIAQknT6XLVQGFKwIy/Ux7lvM2MNap5dt0T7gA==", 246 | "requires": { 247 | "async": "^1.5.0", 248 | "express-unless": "^0.3.0", 249 | "jsonwebtoken": "^8.1.0", 250 | "lodash.set": "^4.0.0" 251 | } 252 | }, 253 | "express-unless": { 254 | "version": "0.3.1", 255 | "resolved": "https://registry.npmjs.org/express-unless/-/express-unless-0.3.1.tgz", 256 | "integrity": "sha1-JVfBRudb65A+LSR/m1ugFFJpbiA=" 257 | }, 258 | "express-validator": { 259 | "version": "5.3.1", 260 | "resolved": "https://registry.npmjs.org/express-validator/-/express-validator-5.3.1.tgz", 261 | "integrity": "sha512-g8xkipBF6VxHbO1+ksC7nxUU7+pWif0+OZXjZTybKJ/V0aTVhuCoHbyhIPgSYVldwQLocGExPtB2pE0DqK4jsw==", 262 | "requires": { 263 | "lodash": "^4.17.10", 264 | "validator": "^10.4.0" 265 | } 266 | }, 267 | "finalhandler": { 268 | "version": "1.1.2", 269 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", 270 | "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", 271 | "requires": { 272 | "debug": "2.6.9", 273 | "encodeurl": "~1.0.2", 274 | "escape-html": "~1.0.3", 275 | "on-finished": "~2.3.0", 276 | "parseurl": "~1.3.3", 277 | "statuses": "~1.5.0", 278 | "unpipe": "~1.0.0" 279 | } 280 | }, 281 | "formidable": { 282 | "version": "1.2.2", 283 | "resolved": "https://registry.npmjs.org/formidable/-/formidable-1.2.2.tgz", 284 | "integrity": "sha512-V8gLm+41I/8kguQ4/o1D3RIHRmhYFG4pnNyonvua+40rqcEmT4+V71yaZ3B457xbbgCsCfjSPi65u/W6vK1U5Q==" 285 | }, 286 | "forwarded": { 287 | "version": "0.1.2", 288 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", 289 | "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" 290 | }, 291 | "fresh": { 292 | "version": "0.5.2", 293 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 294 | "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" 295 | }, 296 | "http-errors": { 297 | "version": "1.7.2", 298 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", 299 | "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", 300 | "requires": { 301 | "depd": "~1.1.2", 302 | "inherits": "2.0.3", 303 | "setprototypeof": "1.1.1", 304 | "statuses": ">= 1.5.0 < 2", 305 | "toidentifier": "1.0.0" 306 | } 307 | }, 308 | "iconv-lite": { 309 | "version": "0.4.24", 310 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 311 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 312 | "requires": { 313 | "safer-buffer": ">= 2.1.2 < 3" 314 | } 315 | }, 316 | "inherits": { 317 | "version": "2.0.3", 318 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 319 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" 320 | }, 321 | "ipaddr.js": { 322 | "version": "1.9.1", 323 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 324 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" 325 | }, 326 | "isarray": { 327 | "version": "1.0.0", 328 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 329 | "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" 330 | }, 331 | "joi": { 332 | "version": "17.2.1", 333 | "resolved": "https://registry.npmjs.org/joi/-/joi-17.2.1.tgz", 334 | "integrity": "sha512-YT3/4Ln+5YRpacdmfEfrrKh50/kkgX3LgBltjqnlMPIYiZ4hxXZuVJcxmsvxsdeHg9soZfE3qXxHC2tMpCCBOA==", 335 | "requires": { 336 | "@hapi/address": "^4.1.0", 337 | "@hapi/formula": "^2.0.0", 338 | "@hapi/hoek": "^9.0.0", 339 | "@hapi/pinpoint": "^2.0.0", 340 | "@hapi/topo": "^5.0.0" 341 | } 342 | }, 343 | "jsonwebtoken": { 344 | "version": "8.5.1", 345 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", 346 | "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", 347 | "requires": { 348 | "jws": "^3.2.2", 349 | "lodash.includes": "^4.3.0", 350 | "lodash.isboolean": "^3.0.3", 351 | "lodash.isinteger": "^4.0.4", 352 | "lodash.isnumber": "^3.0.3", 353 | "lodash.isplainobject": "^4.0.6", 354 | "lodash.isstring": "^4.0.1", 355 | "lodash.once": "^4.0.0", 356 | "ms": "^2.1.1", 357 | "semver": "^5.6.0" 358 | }, 359 | "dependencies": { 360 | "ms": { 361 | "version": "2.1.2", 362 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 363 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 364 | } 365 | } 366 | }, 367 | "jwa": { 368 | "version": "1.4.1", 369 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", 370 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", 371 | "requires": { 372 | "buffer-equal-constant-time": "1.0.1", 373 | "ecdsa-sig-formatter": "1.0.11", 374 | "safe-buffer": "^5.0.1" 375 | } 376 | }, 377 | "jws": { 378 | "version": "3.2.2", 379 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", 380 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", 381 | "requires": { 382 | "jwa": "^1.4.1", 383 | "safe-buffer": "^5.0.1" 384 | } 385 | }, 386 | "kareem": { 387 | "version": "2.3.1", 388 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.1.tgz", 389 | "integrity": "sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw==" 390 | }, 391 | "lodash": { 392 | "version": "4.17.20", 393 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.20.tgz", 394 | "integrity": "sha512-PlhdFcillOINfeV7Ni6oF1TAEayyZBoZ8bcshTHqOYJYlrqzRK5hagpagky5o4HfCzzd1TRkXPMFq6cKk9rGmA==" 395 | }, 396 | "lodash.includes": { 397 | "version": "4.3.0", 398 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", 399 | "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=" 400 | }, 401 | "lodash.isboolean": { 402 | "version": "3.0.3", 403 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", 404 | "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" 405 | }, 406 | "lodash.isinteger": { 407 | "version": "4.0.4", 408 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", 409 | "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=" 410 | }, 411 | "lodash.isnumber": { 412 | "version": "3.0.3", 413 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", 414 | "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=" 415 | }, 416 | "lodash.isplainobject": { 417 | "version": "4.0.6", 418 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", 419 | "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" 420 | }, 421 | "lodash.isstring": { 422 | "version": "4.0.1", 423 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", 424 | "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=" 425 | }, 426 | "lodash.once": { 427 | "version": "4.1.1", 428 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", 429 | "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=" 430 | }, 431 | "lodash.set": { 432 | "version": "4.3.2", 433 | "resolved": "https://registry.npmjs.org/lodash.set/-/lodash.set-4.3.2.tgz", 434 | "integrity": "sha1-2HV7HagH3eJIFrDWqEvqGnYjCyM=" 435 | }, 436 | "media-typer": { 437 | "version": "0.3.0", 438 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 439 | "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" 440 | }, 441 | "memory-pager": { 442 | "version": "1.5.0", 443 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", 444 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", 445 | "optional": true 446 | }, 447 | "merge-descriptors": { 448 | "version": "1.0.1", 449 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", 450 | "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" 451 | }, 452 | "methods": { 453 | "version": "1.1.2", 454 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 455 | "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" 456 | }, 457 | "mime": { 458 | "version": "1.6.0", 459 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 460 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" 461 | }, 462 | "mime-db": { 463 | "version": "1.44.0", 464 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", 465 | "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==" 466 | }, 467 | "mime-types": { 468 | "version": "2.1.27", 469 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", 470 | "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", 471 | "requires": { 472 | "mime-db": "1.44.0" 473 | } 474 | }, 475 | "mongodb": { 476 | "version": "3.5.10", 477 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.5.10.tgz", 478 | "integrity": "sha512-p/C48UvTU/dr/PQEDKfb9DsCVDJWXGmdJNFC+u5FPmTQVtog69X6D8vrWHz+sJx1zJnd96sjdh9ueo7bx2ILTw==", 479 | "requires": { 480 | "bl": "^2.2.0", 481 | "bson": "^1.1.4", 482 | "denque": "^1.4.1", 483 | "require_optional": "^1.0.1", 484 | "safe-buffer": "^5.1.2", 485 | "saslprep": "^1.0.0" 486 | } 487 | }, 488 | "mongoose": { 489 | "version": "5.9.28", 490 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.9.28.tgz", 491 | "integrity": "sha512-A8lNRk4eCQDzk+DagSMYdH94LAYrbTK83LgrUlzqdig3YXvizW3DApJqOWQ5DdhuimvsfiD0Z5NTVzXl/rgi2w==", 492 | "requires": { 493 | "bson": "^1.1.4", 494 | "kareem": "2.3.1", 495 | "mongodb": "3.5.10", 496 | "mongoose-legacy-pluralize": "1.0.2", 497 | "mpath": "0.7.0", 498 | "mquery": "3.2.2", 499 | "ms": "2.1.2", 500 | "regexp-clone": "1.0.0", 501 | "safe-buffer": "5.2.1", 502 | "sift": "7.0.1", 503 | "sliced": "1.0.1" 504 | }, 505 | "dependencies": { 506 | "ms": { 507 | "version": "2.1.2", 508 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", 509 | "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" 510 | }, 511 | "safe-buffer": { 512 | "version": "5.2.1", 513 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 514 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" 515 | } 516 | } 517 | }, 518 | "mongoose-legacy-pluralize": { 519 | "version": "1.0.2", 520 | "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz", 521 | "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ==" 522 | }, 523 | "mpath": { 524 | "version": "0.7.0", 525 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.7.0.tgz", 526 | "integrity": "sha512-Aiq04hILxhz1L+f7sjGyn7IxYzWm1zLNNXcfhDtx04kZ2Gk7uvFdgZ8ts1cWa/6d0TQmag2yR8zSGZUmp0tFNg==" 527 | }, 528 | "mquery": { 529 | "version": "3.2.2", 530 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.2.tgz", 531 | "integrity": "sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==", 532 | "requires": { 533 | "bluebird": "3.5.1", 534 | "debug": "3.1.0", 535 | "regexp-clone": "^1.0.0", 536 | "safe-buffer": "5.1.2", 537 | "sliced": "1.0.1" 538 | }, 539 | "dependencies": { 540 | "debug": { 541 | "version": "3.1.0", 542 | "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", 543 | "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", 544 | "requires": { 545 | "ms": "2.0.0" 546 | } 547 | } 548 | } 549 | }, 550 | "ms": { 551 | "version": "2.0.0", 552 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 553 | "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" 554 | }, 555 | "negotiator": { 556 | "version": "0.6.2", 557 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", 558 | "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" 559 | }, 560 | "object-assign": { 561 | "version": "4.1.1", 562 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 563 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" 564 | }, 565 | "on-finished": { 566 | "version": "2.3.0", 567 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 568 | "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", 569 | "requires": { 570 | "ee-first": "1.1.1" 571 | } 572 | }, 573 | "parseurl": { 574 | "version": "1.3.3", 575 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 576 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" 577 | }, 578 | "path-to-regexp": { 579 | "version": "0.1.7", 580 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", 581 | "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" 582 | }, 583 | "process-nextick-args": { 584 | "version": "2.0.1", 585 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 586 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" 587 | }, 588 | "proxy-addr": { 589 | "version": "2.0.6", 590 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", 591 | "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", 592 | "requires": { 593 | "forwarded": "~0.1.2", 594 | "ipaddr.js": "1.9.1" 595 | } 596 | }, 597 | "qs": { 598 | "version": "6.7.0", 599 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", 600 | "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" 601 | }, 602 | "range-parser": { 603 | "version": "1.2.1", 604 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 605 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" 606 | }, 607 | "raw-body": { 608 | "version": "2.4.0", 609 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", 610 | "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", 611 | "requires": { 612 | "bytes": "3.1.0", 613 | "http-errors": "1.7.2", 614 | "iconv-lite": "0.4.24", 615 | "unpipe": "1.0.0" 616 | } 617 | }, 618 | "readable-stream": { 619 | "version": "2.3.7", 620 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", 621 | "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", 622 | "requires": { 623 | "core-util-is": "~1.0.0", 624 | "inherits": "~2.0.3", 625 | "isarray": "~1.0.0", 626 | "process-nextick-args": "~2.0.0", 627 | "safe-buffer": "~5.1.1", 628 | "string_decoder": "~1.1.1", 629 | "util-deprecate": "~1.0.1" 630 | } 631 | }, 632 | "regexp-clone": { 633 | "version": "1.0.0", 634 | "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz", 635 | "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw==" 636 | }, 637 | "require_optional": { 638 | "version": "1.0.1", 639 | "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz", 640 | "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==", 641 | "requires": { 642 | "resolve-from": "^2.0.0", 643 | "semver": "^5.1.0" 644 | } 645 | }, 646 | "resolve-from": { 647 | "version": "2.0.0", 648 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz", 649 | "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c=" 650 | }, 651 | "safe-buffer": { 652 | "version": "5.1.2", 653 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 654 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" 655 | }, 656 | "safer-buffer": { 657 | "version": "2.1.2", 658 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 659 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 660 | }, 661 | "saslprep": { 662 | "version": "1.0.3", 663 | "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz", 664 | "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==", 665 | "optional": true, 666 | "requires": { 667 | "sparse-bitfield": "^3.0.3" 668 | } 669 | }, 670 | "semver": { 671 | "version": "5.7.1", 672 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", 673 | "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" 674 | }, 675 | "send": { 676 | "version": "0.17.1", 677 | "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", 678 | "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", 679 | "requires": { 680 | "debug": "2.6.9", 681 | "depd": "~1.1.2", 682 | "destroy": "~1.0.4", 683 | "encodeurl": "~1.0.2", 684 | "escape-html": "~1.0.3", 685 | "etag": "~1.8.1", 686 | "fresh": "0.5.2", 687 | "http-errors": "~1.7.2", 688 | "mime": "1.6.0", 689 | "ms": "2.1.1", 690 | "on-finished": "~2.3.0", 691 | "range-parser": "~1.2.1", 692 | "statuses": "~1.5.0" 693 | }, 694 | "dependencies": { 695 | "ms": { 696 | "version": "2.1.1", 697 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 698 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" 699 | } 700 | } 701 | }, 702 | "serve-static": { 703 | "version": "1.14.1", 704 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", 705 | "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", 706 | "requires": { 707 | "encodeurl": "~1.0.2", 708 | "escape-html": "~1.0.3", 709 | "parseurl": "~1.3.3", 710 | "send": "0.17.1" 711 | } 712 | }, 713 | "setprototypeof": { 714 | "version": "1.1.1", 715 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", 716 | "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" 717 | }, 718 | "sift": { 719 | "version": "7.0.1", 720 | "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz", 721 | "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g==" 722 | }, 723 | "sliced": { 724 | "version": "1.0.1", 725 | "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz", 726 | "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E=" 727 | }, 728 | "sparse-bitfield": { 729 | "version": "3.0.3", 730 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", 731 | "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=", 732 | "optional": true, 733 | "requires": { 734 | "memory-pager": "^1.0.2" 735 | } 736 | }, 737 | "statuses": { 738 | "version": "1.5.0", 739 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", 740 | "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" 741 | }, 742 | "string_decoder": { 743 | "version": "1.1.1", 744 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 745 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 746 | "requires": { 747 | "safe-buffer": "~5.1.0" 748 | } 749 | }, 750 | "toidentifier": { 751 | "version": "1.0.0", 752 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", 753 | "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" 754 | }, 755 | "type-is": { 756 | "version": "1.6.18", 757 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 758 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 759 | "requires": { 760 | "media-typer": "0.3.0", 761 | "mime-types": "~2.1.24" 762 | } 763 | }, 764 | "unpipe": { 765 | "version": "1.0.0", 766 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 767 | "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" 768 | }, 769 | "util-deprecate": { 770 | "version": "1.0.2", 771 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 772 | "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" 773 | }, 774 | "utils-merge": { 775 | "version": "1.0.1", 776 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 777 | "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" 778 | }, 779 | "uuid": { 780 | "version": "8.3.0", 781 | "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.0.tgz", 782 | "integrity": "sha512-fX6Z5o4m6XsXBdli9g7DtWgAx+osMsRRZFKma1mIUsLCz6vRvv+pz5VNbyu9UEDzpMWulZfvpgb/cmDXVulYFQ==" 783 | }, 784 | "validator": { 785 | "version": "10.11.0", 786 | "resolved": "https://registry.npmjs.org/validator/-/validator-10.11.0.tgz", 787 | "integrity": "sha512-X/p3UZerAIsbBfN/IwahhYaBbY68EN/UQBWHtsbXGT5bfrH/p4NQzUCG1kF/rtKaNpnJ7jAu6NGTdSNtyNIXMw==" 788 | }, 789 | "vary": { 790 | "version": "1.1.2", 791 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 792 | "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" 793 | } 794 | } 795 | } 796 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "ecommerce", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "test": "echo \"Error: no test specified\" && exit 1" 8 | }, 9 | "keywords": [], 10 | "author": "", 11 | "license": "ISC", 12 | "dependencies": { 13 | "cookie-parser": "^1.4.5", 14 | "cors": "^2.8.5", 15 | "dotenv": "^8.2.0", 16 | "express": "^4.17.1", 17 | "express-jwt": "^6.0.0", 18 | "express-validator": "^5.3.1", 19 | "formidable": "^1.2.2", 20 | "joi": "^17.2.1", 21 | "jsonwebtoken": "^8.5.1", 22 | "lodash": "^4.17.20", 23 | "mongoose": "^5.9.28", 24 | "uuid": "^8.3.0" 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /routes/auth.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const { salam, signup, signin, signout } = require('../controllers/authController') 3 | const { userSignUpValidator } = require('../middlewares/userValidator') 4 | const { requireSignIn } = require('../middlewares/auth'); 5 | const router = express.Router(); 6 | 7 | router.get('/', salam); 8 | 9 | router.post('/signup', userSignUpValidator, signup) 10 | router.post('/signin', signin) 11 | router.get('/signout', signout) 12 | 13 | router.get("/hello", requireSignIn, (req, res) => { 14 | res.send('hello there'); 15 | }) 16 | 17 | module.exports = router; -------------------------------------------------------------------------------- /routes/categories.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | 3 | const { userById } = require('../middlewares/user'); 4 | 5 | const router = express.Router(); 6 | 7 | const { 8 | allCategories, 9 | createCategory, 10 | categoryId, 11 | showCategory, 12 | updateCategory, 13 | deleteCategory 14 | } = require('../controllers/categoryController') 15 | 16 | const { requireSignIn, isAuth, isAdmin } = require('../middlewares/auth'); 17 | 18 | router.get('/', allCategories); 19 | 20 | router.get('/:categoryId', showCategory); 21 | 22 | router.post('/create/:userId', [requireSignIn, isAuth, isAdmin], createCategory); 23 | 24 | router.put('/:categoryId/:userId', [requireSignIn, isAuth, isAdmin], updateCategory); 25 | 26 | router.delete('/:categoryId/:userId', [requireSignIn, isAuth, isAdmin], deleteCategory); 27 | 28 | 29 | router.param('userId', userById) 30 | 31 | router.param('categoryId', categoryId); 32 | 33 | module.exports = router; -------------------------------------------------------------------------------- /routes/products.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | 3 | const { userById } = require('../middlewares/user'); 4 | 5 | const router = express.Router(); 6 | 7 | const { 8 | allProducts, 9 | relatedProduct, 10 | createProduct, 11 | showProduct, 12 | productById, 13 | removeProduct, 14 | updateProduct, 15 | SearchProduct, 16 | photoProduct 17 | } = require('../controllers/productController') 18 | 19 | const { requireSignIn, isAuth, isAdmin } = require('../middlewares/auth'); 20 | 21 | router.get('/', allProducts); 22 | 23 | router.get('/:productId', showProduct); 24 | 25 | router.get('/related/:productId', relatedProduct); 26 | 27 | router.post('/create/:userId', [requireSignIn, isAuth, isAdmin], createProduct); 28 | 29 | router.post('/search', SearchProduct); 30 | 31 | router.get('/photo/:productId', photoProduct); 32 | 33 | router.put('/:productId/:userId', [requireSignIn, isAuth, isAdmin], updateProduct) 34 | 35 | router.delete('/:productId/:userId', [requireSignIn, isAuth, isAdmin], removeProduct) 36 | 37 | router.param('userId', userById) 38 | router.param('productId', productById) 39 | 40 | module.exports = router; 41 | 42 | 43 | -------------------------------------------------------------------------------- /routes/users.js: -------------------------------------------------------------------------------- 1 | const express = require('express'); 2 | const { getOneUser, updateOneUser } = require('../controllers/userController') 3 | const { userById } = require('../middlewares/user'); 4 | const { requireSignIn, isAuth, isAdmin } = require('../middlewares/auth') 5 | 6 | const router = express.Router(); 7 | 8 | router.get('/:userId', requireSignIn, isAuth, getOneUser) 9 | 10 | router.put('/:userId', requireSignIn, isAuth, updateOneUser) 11 | 12 | router.param('userId', userById) 13 | 14 | 15 | module.exports = router; --------------------------------------------------------------------------------