├── .gitignore ├── README.md ├── index.js ├── middleware └── auth.js ├── models ├── Admin.js ├── BookReview.js ├── Message.js └── Project.js ├── package-lock.json ├── package.json ├── public ├── css │ └── style.css └── js │ └── main.js ├── routes ├── admin.routes.js ├── bookReview.routes.js ├── message.routes.js └── project.routes.js └── views └── index.pug /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | .env -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Portfolio Website Project - README 2 | 3 | ## Overview 4 | 5 | This project is a full-stack web application designed to showcase a developer's portfolio. It includes features for displaying projects, book reviews, and user messages, with an admin panel for content management. 6 | 7 | ## Features 8 | 9 | - **Projects:** 10 | - Display a list of projects with titles, descriptions, images, and links. 11 | - View individual project details. 12 | - Admin users can create, update, and delete projects. 13 | - **Book Reviews:** 14 | - Display book reviews with information fetched from the Google Books API. 15 | - Admin users can add, edit, and remove book reviews. 16 | - **Messages:** 17 | - Allow visitors to send messages (e.g., contact form, questions). 18 | - Display testimonials. 19 | - Admin users can view and manage all messages. 20 | - **Admin Panel:** 21 | - Secure authentication for admin users. 22 | - Dashboard for managing projects, book reviews, and messages. 23 | - **User Authentication:** 24 | - User registration and login. 25 | 26 | ## Technologies Used 27 | 28 | - **Backend:** 29 | - Node.js 30 | - Express.js 31 | - MongoDB 32 | - Mongoose 33 | - jsonwebtoken 34 | - bcrypt 35 | - axios 36 | - multer 37 | 38 | ## Backend Setup 39 | 40 | ### Prerequisites 41 | 42 | - Node.js and npm installed 43 | - MongoDB installed and running 44 | - Google Books API Key (If using book review functionality) 45 | 46 | ### Installation 47 | 48 | 1. Clone the repository. 49 | 2. Navigate to the `backend` directory. 50 | 3. Install dependencies: `npm install` 51 | 4. Create a `.env` file in the `backend` directory and add the following environment variables: 52 | - `MONGODB_URI`: Your MongoDB connection string 53 | - `JWT_SECRET`: A secret key for JWT authentication (generate a strong key) 54 | - `GOOGLE_BOOKS_API_KEY`: Your Google Books API key (if applicable) 55 | - `PORT`: The port the server will listen on (e.g., 5000) 56 | 5. Start the server: `npm start` 57 | 58 | ### Database 59 | 60 | - The application uses MongoDB to store data. 61 | - Mongoose is used as the ODM (Object Data Modeling) library. 62 | - Schemas are defined in the `backend/models` directory: 63 | - `Admin.js`: Admin user schema 64 | - `BookReview.js`: Book review schema 65 | - `Message.js`: Message schema 66 | - `Project.js`: Project schema 67 | - `User.js`: User schema 68 | 69 | ### API Endpoints 70 | 71 | - The API routes are defined in the `backend/routes` directory: 72 | - `admin.routes.js`: Admin authentication and registration 73 | - `bookReview.routes.js`: CRUD operations for book reviews 74 | - `message.routes.js`: CRUD operations for messages, handling testimonials 75 | - `project.routes.js`: CRUD operations for projects 76 | - `user.js`: CRUD operations for users 77 | 78 | ### Middleware 79 | 80 | - `backend/middleware/auth.js`: Contains authentication middleware for admin routes, using JWT. 81 | 82 | ## Frontend Setup 83 | 84 | ### Prerequisites 85 | 86 | - Node.js and npm installed 87 | 88 | ### Installation 89 | 90 | 1. Navigate to the `frontend` directory. 91 | 2. Install dependencies: `npm install` 92 | 3. Start the development server: `npm start` 93 | 94 | ### Routing 95 | 96 | - React Router DOM is used for frontend routing. 97 | - `createBrowserRouter` is used to define the routes. 98 | - The main routing logic is typically found in `index.js` or `App.js`. 99 | 100 | ### Components 101 | 102 | - The frontend is structured using React components. 103 | - A layout component (`App.js`) provides the basic structure (header, main, footer) 104 | - (You'll need to create page components for home, projects, book reviews, contact, admin dashboard, etc.) 105 | 106 | ## Important Notes 107 | 108 | - **Security:** 109 | - Password hashing is implemented using bcrypt. 110 | - JWT is used for authentication. 111 | - Proper authorization is crucial to protect admin routes. 112 | - Sanitize user inputs to prevent vulnerabilities (e.g., XSS). 113 | - **Error Handling:** Implement robust error handling in both the backend and frontend. 114 | - **Data Validation:** Validate data on both the client-side and server-side. 115 | - **File Uploads:** 116 | - Multer is used to handle file uploads in the backend. 117 | - The `public` directory is used to serve static files. 118 | - (This part might be removed or modified based on the current implementation) 119 | - **API Keys:** 120 | - Protect your API keys (e.g., Google Books API key) and store them securely as environment variables. 121 | - **Deployment:** 122 | - You'll need to deploy both the backend and frontend separately. 123 | - Consider using platforms like Heroku, Netlify, or Vercel for deployment. 124 | 125 | ## Future Improvements 126 | 127 | - Implement more advanced admin panel features. 128 | - Add user roles and permissions. 129 | - Implement search and filtering. 130 | - Add pagination for large datasets. 131 | - Improve styling and UI design. 132 | - Write tests for backend API and frontend components. 133 | -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | import morgan from "morgan"; 3 | import helmet from "helmet"; 4 | import dotenv from "dotenv"; 5 | import cors from "cors"; 6 | import mongoose from "mongoose"; 7 | 8 | // Routers 9 | import adminRouter from "./routes/admin.routes.js"; 10 | import projectRouter from "./routes/project.routes.js"; 11 | import messageRouter from "./routes/message.routes.js"; 12 | import bookReviewRouter from "./routes/bookReview.routes.js"; 13 | 14 | dotenv.config(); 15 | // console.log(process.env.MONGODB_URI); 16 | 17 | // Connect to MongoDB 18 | // https://mongoosejs.com/docs/guide.html#indexes 19 | await mongoose 20 | .connect(process.env.MONGODB_URI) 21 | .then(() => console.log("Connected to MongoDB")) 22 | .catch((e) => console.error(e)); 23 | 24 | const PORT = process.env.PORT; 25 | 26 | const app = express(); 27 | 28 | // View Engine 29 | app.set("views", "./views"); 30 | app.set("view engine", "pug"); 31 | 32 | // Middlewares 33 | app.use(express.static("./public")); 34 | app.use(express.json()); 35 | app.use(express.urlencoded({ extended: true })); 36 | app.use(morgan("dev")); 37 | app.use(helmet()); 38 | app.use(cors()); 39 | 40 | // Routes 41 | app.get("/", (req, res) => { 42 | res.render("index"); 43 | }); 44 | 45 | // API Routes 46 | app.use("/api/admin", adminRouter); 47 | app.use("/api/project", projectRouter); 48 | app.use("/api/book", bookReviewRouter); 49 | app.use("/api/message", messageRouter); 50 | 51 | // Global error handling 52 | app.use((err, req, res, next) => { 53 | console.error(err); 54 | res.status(500).send("Seems like we messed up somewhere..."); 55 | }); 56 | 57 | app.listen(PORT, () => console.log(`Server is running on port: ${PORT}`)); 58 | -------------------------------------------------------------------------------- /middleware/auth.js: -------------------------------------------------------------------------------- 1 | import jwt from "jsonwebtoken"; 2 | import Admin from "../models/Admin.js"; 3 | 4 | const authenticateAdmin = async (req, res, next) => { 5 | let token; 6 | 7 | if ( 8 | req.headers.authorization && 9 | req.headers.authorization.startsWith("Bearer") 10 | ) { 11 | try { 12 | token = req.headers.authorization.split(" ")[1]; 13 | const decoded = jwt.verify(token, process.env.JWT_SECRET); 14 | req.admin = await Admin.findById(decoded.id).select("-password"); 15 | 16 | if (!req.admin) { 17 | return res 18 | .status(401) 19 | .json({ message: "Not authorized, admin not found" }); 20 | } 21 | 22 | next(); 23 | } catch (error) { 24 | console.error(error); 25 | res.status(401).json({ message: "Not authorized, token failed" }); 26 | } 27 | } 28 | 29 | if (!token) { 30 | res.status(401).json({ message: "Not authorized, no token" }); 31 | } 32 | }; 33 | 34 | export { authenticateAdmin }; 35 | -------------------------------------------------------------------------------- /models/Admin.js: -------------------------------------------------------------------------------- 1 | import mongoose from "mongoose"; 2 | import bcrypt from "bcrypt"; 3 | 4 | const adminSchema = new mongoose.Schema({ 5 | username: { type: String, required: true, unique: true }, 6 | password: { type: String, required: true }, 7 | }); 8 | 9 | // Hash the password before saving 10 | adminSchema.pre("save", async function (next) { 11 | if (!this.isModified("password")) return next(); 12 | this.password = await bcrypt.hash(this.password, 10); 13 | next(); 14 | }); 15 | 16 | adminSchema.methods.comparePassword = async function (password) { 17 | return bcrypt.compare(password, this.password); 18 | }; 19 | 20 | const Admin = mongoose.model("Admin", adminSchema); 21 | 22 | export default Admin; 23 | -------------------------------------------------------------------------------- /models/BookReview.js: -------------------------------------------------------------------------------- 1 | import mongoose from "mongoose"; 2 | 3 | const bookReviewSchema = new mongoose.Schema({ 4 | bookId: { 5 | type: String, // Google Books API book ID 6 | required: true, 7 | unique: true, 8 | }, 9 | title: { 10 | type: String, 11 | required: true, 12 | }, 13 | subTitle: { 14 | type: String, 15 | required: true, 16 | }, 17 | authors: [{ type: String, required: true }], 18 | thumbnail: { 19 | type: String, 20 | }, 21 | genres: [{ type: String }], // Array for book genres 22 | descriptionShort: { 23 | type: String, 24 | required: true, 25 | }, 26 | howChangedMe: { 27 | type: String, 28 | required: true, 29 | }, 30 | whoShouldRead: { 31 | type: String, 32 | required: true, 33 | }, 34 | impressions: { 35 | type: String, 36 | required: true, 37 | }, 38 | rating: { 39 | type: Number, 40 | min: 1, 41 | max: 5, 42 | }, 43 | readAt: { 44 | type: Date, 45 | default: Date.now, 46 | }, 47 | updatedAt: { 48 | type: Date, 49 | default: Date.now, 50 | }, 51 | }); 52 | 53 | const BookReview = mongoose.model("BookReview", bookReviewSchema); 54 | 55 | export default BookReview; 56 | -------------------------------------------------------------------------------- /models/Message.js: -------------------------------------------------------------------------------- 1 | import mongoose from "mongoose"; 2 | 3 | const messageSchema = new mongoose.Schema({ 4 | type: { 5 | type: String, 6 | enum: ["question", "testimonial", "contact", "other"], 7 | required: true, 8 | }, 9 | content: { 10 | type: String, 11 | required: true, 12 | }, 13 | name: { 14 | type: String, 15 | required: true, 16 | trim: true, 17 | }, 18 | email: { 19 | type: String, 20 | trim: true, 21 | }, 22 | authorTitle: { 23 | type: String, 24 | trim: true, 25 | }, 26 | createdAt: { 27 | type: Date, 28 | default: Date.now, 29 | }, 30 | isRead: { 31 | type: Boolean, 32 | default: false, 33 | }, 34 | isResponded: { 35 | type: Boolean, 36 | default: false, 37 | }, 38 | }); 39 | 40 | const Message = mongoose.model("Message", messageSchema); 41 | 42 | export default Message; 43 | -------------------------------------------------------------------------------- /models/Project.js: -------------------------------------------------------------------------------- 1 | import mongoose from "mongoose"; 2 | 3 | const projectSchema = new mongoose.Schema({ 4 | title: { 5 | type: String, 6 | required: true, 7 | trim: true, 8 | }, 9 | description: { 10 | type: String, 11 | required: true, 12 | }, 13 | imageUrl: { 14 | type: String, 15 | }, 16 | link: { 17 | type: String, 18 | }, 19 | technologies: [ 20 | { 21 | type: String, 22 | trim: true, 23 | }, 24 | ], 25 | createdAt: { 26 | type: Date, 27 | default: Date.now, 28 | }, 29 | updatedAt: { 30 | type: Date, 31 | default: Date.now, 32 | }, 33 | }); 34 | 35 | const Project = mongoose.model("Project", projectSchema); 36 | 37 | export default Project; 38 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "backend-template", 3 | "version": "1.0.0", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "backend-template", 9 | "version": "1.0.0", 10 | "license": "ISC", 11 | "dependencies": { 12 | "axios": "^1.8.4", 13 | "bcrypt": "^5.1.1", 14 | "cors": "^2.8.5", 15 | "dotenv": "^16.4.7", 16 | "express": "^4.21.2", 17 | "helmet": "^8.0.0", 18 | "jsonwebtoken": "^9.0.2", 19 | "mongoose": "^8.12.1", 20 | "pug": "^3.0.3" 21 | }, 22 | "devDependencies": { 23 | "morgan": "^1.10.0", 24 | "nodemon": "^3.1.9" 25 | } 26 | }, 27 | "node_modules/@babel/helper-string-parser": { 28 | "version": "7.25.9", 29 | "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.25.9.tgz", 30 | "integrity": "sha512-4A/SCr/2KLd5jrtOMFzaKjVtAei3+2r/NChoBNoZ3EyP/+GlhoaEGoWOZUmFmoITP7zOJyHIMm+DYRd8o3PvHA==", 31 | "engines": { 32 | "node": ">=6.9.0" 33 | } 34 | }, 35 | "node_modules/@babel/helper-validator-identifier": { 36 | "version": "7.25.9", 37 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.25.9.tgz", 38 | "integrity": "sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ==", 39 | "engines": { 40 | "node": ">=6.9.0" 41 | } 42 | }, 43 | "node_modules/@babel/parser": { 44 | "version": "7.26.10", 45 | "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.26.10.tgz", 46 | "integrity": "sha512-6aQR2zGE/QFi8JpDLjUZEPYOs7+mhKXm86VaKFiLP35JQwQb6bwUE+XbvkH0EptsYhbNBSUGaUBLKqxH1xSgsA==", 47 | "dependencies": { 48 | "@babel/types": "^7.26.10" 49 | }, 50 | "bin": { 51 | "parser": "bin/babel-parser.js" 52 | }, 53 | "engines": { 54 | "node": ">=6.0.0" 55 | } 56 | }, 57 | "node_modules/@babel/types": { 58 | "version": "7.26.10", 59 | "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.26.10.tgz", 60 | "integrity": "sha512-emqcG3vHrpxUKTrxcblR36dcrcoRDvKmnL/dCL6ZsHaShW80qxCAcNhzQZrpeM765VzEos+xOi4s+r4IXzTwdQ==", 61 | "dependencies": { 62 | "@babel/helper-string-parser": "^7.25.9", 63 | "@babel/helper-validator-identifier": "^7.25.9" 64 | }, 65 | "engines": { 66 | "node": ">=6.9.0" 67 | } 68 | }, 69 | "node_modules/@mapbox/node-pre-gyp": { 70 | "version": "1.0.11", 71 | "resolved": "https://registry.npmjs.org/@mapbox/node-pre-gyp/-/node-pre-gyp-1.0.11.tgz", 72 | "integrity": "sha512-Yhlar6v9WQgUp/He7BdgzOz8lqMQ8sU+jkCq7Wx8Myc5YFJLbEe7lgui/V7G1qB1DJykHSGwreceSaD60Y0PUQ==", 73 | "license": "BSD-3-Clause", 74 | "dependencies": { 75 | "detect-libc": "^2.0.0", 76 | "https-proxy-agent": "^5.0.0", 77 | "make-dir": "^3.1.0", 78 | "node-fetch": "^2.6.7", 79 | "nopt": "^5.0.0", 80 | "npmlog": "^5.0.1", 81 | "rimraf": "^3.0.2", 82 | "semver": "^7.3.5", 83 | "tar": "^6.1.11" 84 | }, 85 | "bin": { 86 | "node-pre-gyp": "bin/node-pre-gyp" 87 | } 88 | }, 89 | "node_modules/@mongodb-js/saslprep": { 90 | "version": "1.2.0", 91 | "resolved": "https://registry.npmjs.org/@mongodb-js/saslprep/-/saslprep-1.2.0.tgz", 92 | "integrity": "sha512-+ywrb0AqkfaYuhHs6LxKWgqbh3I72EpEgESCw37o+9qPx9WTCkgDm2B+eMrwehGtHBWHFU4GXvnSCNiFhhausg==", 93 | "dependencies": { 94 | "sparse-bitfield": "^3.0.3" 95 | } 96 | }, 97 | "node_modules/@types/webidl-conversions": { 98 | "version": "7.0.3", 99 | "resolved": "https://registry.npmjs.org/@types/webidl-conversions/-/webidl-conversions-7.0.3.tgz", 100 | "integrity": "sha512-CiJJvcRtIgzadHCYXw7dqEnMNRjhGZlYK05Mj9OyktqV8uVT8fD2BFOB7S1uwBE3Kj2Z+4UyPmFw/Ixgw/LAlA==" 101 | }, 102 | "node_modules/@types/whatwg-url": { 103 | "version": "11.0.5", 104 | "resolved": "https://registry.npmjs.org/@types/whatwg-url/-/whatwg-url-11.0.5.tgz", 105 | "integrity": "sha512-coYR071JRaHa+xoEvvYqvnIHaVqaYrLPbsufM9BF63HkwI5Lgmy2QR8Q5K/lYDYo5AK82wOvSOS0UsLTpTG7uQ==", 106 | "dependencies": { 107 | "@types/webidl-conversions": "*" 108 | } 109 | }, 110 | "node_modules/abbrev": { 111 | "version": "1.1.1", 112 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", 113 | "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", 114 | "license": "ISC" 115 | }, 116 | "node_modules/accepts": { 117 | "version": "1.3.8", 118 | "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", 119 | "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", 120 | "dependencies": { 121 | "mime-types": "~2.1.34", 122 | "negotiator": "0.6.3" 123 | }, 124 | "engines": { 125 | "node": ">= 0.6" 126 | } 127 | }, 128 | "node_modules/acorn": { 129 | "version": "7.4.1", 130 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", 131 | "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", 132 | "bin": { 133 | "acorn": "bin/acorn" 134 | }, 135 | "engines": { 136 | "node": ">=0.4.0" 137 | } 138 | }, 139 | "node_modules/agent-base": { 140 | "version": "6.0.2", 141 | "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", 142 | "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", 143 | "license": "MIT", 144 | "dependencies": { 145 | "debug": "4" 146 | }, 147 | "engines": { 148 | "node": ">= 6.0.0" 149 | } 150 | }, 151 | "node_modules/agent-base/node_modules/debug": { 152 | "version": "4.4.0", 153 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", 154 | "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", 155 | "license": "MIT", 156 | "dependencies": { 157 | "ms": "^2.1.3" 158 | }, 159 | "engines": { 160 | "node": ">=6.0" 161 | }, 162 | "peerDependenciesMeta": { 163 | "supports-color": { 164 | "optional": true 165 | } 166 | } 167 | }, 168 | "node_modules/agent-base/node_modules/ms": { 169 | "version": "2.1.3", 170 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 171 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 172 | "license": "MIT" 173 | }, 174 | "node_modules/ansi-regex": { 175 | "version": "5.0.1", 176 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 177 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 178 | "license": "MIT", 179 | "engines": { 180 | "node": ">=8" 181 | } 182 | }, 183 | "node_modules/anymatch": { 184 | "version": "3.1.3", 185 | "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", 186 | "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", 187 | "dev": true, 188 | "dependencies": { 189 | "normalize-path": "^3.0.0", 190 | "picomatch": "^2.0.4" 191 | }, 192 | "engines": { 193 | "node": ">= 8" 194 | } 195 | }, 196 | "node_modules/aproba": { 197 | "version": "2.0.0", 198 | "resolved": "https://registry.npmjs.org/aproba/-/aproba-2.0.0.tgz", 199 | "integrity": "sha512-lYe4Gx7QT+MKGbDsA+Z+he/Wtef0BiwDOlK/XkBrdfsh9J/jPPXbX0tE9x9cl27Tmu5gg3QUbUrQYa/y+KOHPQ==", 200 | "license": "ISC" 201 | }, 202 | "node_modules/are-we-there-yet": { 203 | "version": "2.0.0", 204 | "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-2.0.0.tgz", 205 | "integrity": "sha512-Ci/qENmwHnsYo9xKIcUJN5LeDKdJ6R1Z1j9V/J5wyq8nh/mYPEpIKJbBZXtZjG04HiK7zV/p6Vs9952MrMeUIw==", 206 | "deprecated": "This package is no longer supported.", 207 | "license": "ISC", 208 | "dependencies": { 209 | "delegates": "^1.0.0", 210 | "readable-stream": "^3.6.0" 211 | }, 212 | "engines": { 213 | "node": ">=10" 214 | } 215 | }, 216 | "node_modules/array-flatten": { 217 | "version": "1.1.1", 218 | "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", 219 | "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==" 220 | }, 221 | "node_modules/asap": { 222 | "version": "2.0.6", 223 | "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", 224 | "integrity": "sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==" 225 | }, 226 | "node_modules/assert-never": { 227 | "version": "1.4.0", 228 | "resolved": "https://registry.npmjs.org/assert-never/-/assert-never-1.4.0.tgz", 229 | "integrity": "sha512-5oJg84os6NMQNl27T9LnZkvvqzvAnHu03ShCnoj6bsJwS7L8AO4lf+C/XjK/nvzEqQB744moC6V128RucQd1jA==" 230 | }, 231 | "node_modules/asynckit": { 232 | "version": "0.4.0", 233 | "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", 234 | "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", 235 | "license": "MIT" 236 | }, 237 | "node_modules/axios": { 238 | "version": "1.8.4", 239 | "resolved": "https://registry.npmjs.org/axios/-/axios-1.8.4.tgz", 240 | "integrity": "sha512-eBSYY4Y68NNlHbHBMdeDmKNtDgXWhQsJcGqzO3iLUM0GraQFSS9cVgPX5I9b3lbdFKyYoAEGAZF1DwhTaljNAw==", 241 | "license": "MIT", 242 | "dependencies": { 243 | "follow-redirects": "^1.15.6", 244 | "form-data": "^4.0.0", 245 | "proxy-from-env": "^1.1.0" 246 | } 247 | }, 248 | "node_modules/babel-walk": { 249 | "version": "3.0.0-canary-5", 250 | "resolved": "https://registry.npmjs.org/babel-walk/-/babel-walk-3.0.0-canary-5.tgz", 251 | "integrity": "sha512-GAwkz0AihzY5bkwIY5QDR+LvsRQgB/B+1foMPvi0FZPMl5fjD7ICiznUiBdLYMH1QYe6vqu4gWYytZOccLouFw==", 252 | "dependencies": { 253 | "@babel/types": "^7.9.6" 254 | }, 255 | "engines": { 256 | "node": ">= 10.0.0" 257 | } 258 | }, 259 | "node_modules/balanced-match": { 260 | "version": "1.0.2", 261 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 262 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" 263 | }, 264 | "node_modules/basic-auth": { 265 | "version": "2.0.1", 266 | "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz", 267 | "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==", 268 | "dev": true, 269 | "dependencies": { 270 | "safe-buffer": "5.1.2" 271 | }, 272 | "engines": { 273 | "node": ">= 0.8" 274 | } 275 | }, 276 | "node_modules/basic-auth/node_modules/safe-buffer": { 277 | "version": "5.1.2", 278 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 279 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", 280 | "dev": true 281 | }, 282 | "node_modules/bcrypt": { 283 | "version": "5.1.1", 284 | "resolved": "https://registry.npmjs.org/bcrypt/-/bcrypt-5.1.1.tgz", 285 | "integrity": "sha512-AGBHOG5hPYZ5Xl9KXzU5iKq9516yEmvCKDg3ecP5kX2aB6UqTeXZxk2ELnDgDm6BQSMlLt9rDB4LoSMx0rYwww==", 286 | "hasInstallScript": true, 287 | "license": "MIT", 288 | "dependencies": { 289 | "@mapbox/node-pre-gyp": "^1.0.11", 290 | "node-addon-api": "^5.0.0" 291 | }, 292 | "engines": { 293 | "node": ">= 10.0.0" 294 | } 295 | }, 296 | "node_modules/binary-extensions": { 297 | "version": "2.3.0", 298 | "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", 299 | "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", 300 | "dev": true, 301 | "engines": { 302 | "node": ">=8" 303 | }, 304 | "funding": { 305 | "url": "https://github.com/sponsors/sindresorhus" 306 | } 307 | }, 308 | "node_modules/body-parser": { 309 | "version": "1.20.3", 310 | "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.3.tgz", 311 | "integrity": "sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g==", 312 | "dependencies": { 313 | "bytes": "3.1.2", 314 | "content-type": "~1.0.5", 315 | "debug": "2.6.9", 316 | "depd": "2.0.0", 317 | "destroy": "1.2.0", 318 | "http-errors": "2.0.0", 319 | "iconv-lite": "0.4.24", 320 | "on-finished": "2.4.1", 321 | "qs": "6.13.0", 322 | "raw-body": "2.5.2", 323 | "type-is": "~1.6.18", 324 | "unpipe": "1.0.0" 325 | }, 326 | "engines": { 327 | "node": ">= 0.8", 328 | "npm": "1.2.8000 || >= 1.4.16" 329 | } 330 | }, 331 | "node_modules/brace-expansion": { 332 | "version": "1.1.11", 333 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 334 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 335 | "dependencies": { 336 | "balanced-match": "^1.0.0", 337 | "concat-map": "0.0.1" 338 | } 339 | }, 340 | "node_modules/braces": { 341 | "version": "3.0.3", 342 | "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", 343 | "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", 344 | "dev": true, 345 | "dependencies": { 346 | "fill-range": "^7.1.1" 347 | }, 348 | "engines": { 349 | "node": ">=8" 350 | } 351 | }, 352 | "node_modules/bson": { 353 | "version": "6.10.3", 354 | "resolved": "https://registry.npmjs.org/bson/-/bson-6.10.3.tgz", 355 | "integrity": "sha512-MTxGsqgYTwfshYWTRdmZRC+M7FnG1b4y7RO7p2k3X24Wq0yv1m77Wsj0BzlPzd/IowgESfsruQCUToa7vbOpPQ==", 356 | "engines": { 357 | "node": ">=16.20.1" 358 | } 359 | }, 360 | "node_modules/buffer-equal-constant-time": { 361 | "version": "1.0.1", 362 | "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", 363 | "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==", 364 | "license": "BSD-3-Clause" 365 | }, 366 | "node_modules/bytes": { 367 | "version": "3.1.2", 368 | "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", 369 | "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", 370 | "engines": { 371 | "node": ">= 0.8" 372 | } 373 | }, 374 | "node_modules/call-bind-apply-helpers": { 375 | "version": "1.0.2", 376 | "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", 377 | "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", 378 | "dependencies": { 379 | "es-errors": "^1.3.0", 380 | "function-bind": "^1.1.2" 381 | }, 382 | "engines": { 383 | "node": ">= 0.4" 384 | } 385 | }, 386 | "node_modules/call-bound": { 387 | "version": "1.0.4", 388 | "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", 389 | "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", 390 | "dependencies": { 391 | "call-bind-apply-helpers": "^1.0.2", 392 | "get-intrinsic": "^1.3.0" 393 | }, 394 | "engines": { 395 | "node": ">= 0.4" 396 | }, 397 | "funding": { 398 | "url": "https://github.com/sponsors/ljharb" 399 | } 400 | }, 401 | "node_modules/character-parser": { 402 | "version": "2.2.0", 403 | "resolved": "https://registry.npmjs.org/character-parser/-/character-parser-2.2.0.tgz", 404 | "integrity": "sha512-+UqJQjFEFaTAs3bNsF2j2kEN1baG/zghZbdqoYEDxGZtJo9LBzl1A+m0D4n3qKx8N2FNv8/Xp6yV9mQmBuptaw==", 405 | "dependencies": { 406 | "is-regex": "^1.0.3" 407 | } 408 | }, 409 | "node_modules/chokidar": { 410 | "version": "3.6.0", 411 | "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", 412 | "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==", 413 | "dev": true, 414 | "dependencies": { 415 | "anymatch": "~3.1.2", 416 | "braces": "~3.0.2", 417 | "glob-parent": "~5.1.2", 418 | "is-binary-path": "~2.1.0", 419 | "is-glob": "~4.0.1", 420 | "normalize-path": "~3.0.0", 421 | "readdirp": "~3.6.0" 422 | }, 423 | "engines": { 424 | "node": ">= 8.10.0" 425 | }, 426 | "funding": { 427 | "url": "https://paulmillr.com/funding/" 428 | }, 429 | "optionalDependencies": { 430 | "fsevents": "~2.3.2" 431 | } 432 | }, 433 | "node_modules/chownr": { 434 | "version": "2.0.0", 435 | "resolved": "https://registry.npmjs.org/chownr/-/chownr-2.0.0.tgz", 436 | "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", 437 | "license": "ISC", 438 | "engines": { 439 | "node": ">=10" 440 | } 441 | }, 442 | "node_modules/color-support": { 443 | "version": "1.1.3", 444 | "resolved": "https://registry.npmjs.org/color-support/-/color-support-1.1.3.tgz", 445 | "integrity": "sha512-qiBjkpbMLO/HL68y+lh4q0/O1MZFj2RX6X/KmMa3+gJD3z+WwI1ZzDHysvqHGS3mP6mznPckpXmw1nI9cJjyRg==", 446 | "license": "ISC", 447 | "bin": { 448 | "color-support": "bin.js" 449 | } 450 | }, 451 | "node_modules/combined-stream": { 452 | "version": "1.0.8", 453 | "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", 454 | "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", 455 | "license": "MIT", 456 | "dependencies": { 457 | "delayed-stream": "~1.0.0" 458 | }, 459 | "engines": { 460 | "node": ">= 0.8" 461 | } 462 | }, 463 | "node_modules/concat-map": { 464 | "version": "0.0.1", 465 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 466 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==" 467 | }, 468 | "node_modules/console-control-strings": { 469 | "version": "1.1.0", 470 | "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", 471 | "integrity": "sha512-ty/fTekppD2fIwRvnZAVdeOiGd1c7YXEixbgJTNzqcxJWKQnjJ/V1bNEEE6hygpM3WjwHFUVK6HTjWSzV4a8sQ==", 472 | "license": "ISC" 473 | }, 474 | "node_modules/constantinople": { 475 | "version": "4.0.1", 476 | "resolved": "https://registry.npmjs.org/constantinople/-/constantinople-4.0.1.tgz", 477 | "integrity": "sha512-vCrqcSIq4//Gx74TXXCGnHpulY1dskqLTFGDmhrGxzeXL8lF8kvXv6mpNWlJj1uD4DW23D4ljAqbY4RRaaUZIw==", 478 | "dependencies": { 479 | "@babel/parser": "^7.6.0", 480 | "@babel/types": "^7.6.1" 481 | } 482 | }, 483 | "node_modules/content-disposition": { 484 | "version": "0.5.4", 485 | "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", 486 | "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", 487 | "dependencies": { 488 | "safe-buffer": "5.2.1" 489 | }, 490 | "engines": { 491 | "node": ">= 0.6" 492 | } 493 | }, 494 | "node_modules/content-type": { 495 | "version": "1.0.5", 496 | "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", 497 | "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", 498 | "engines": { 499 | "node": ">= 0.6" 500 | } 501 | }, 502 | "node_modules/cookie": { 503 | "version": "0.7.1", 504 | "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.1.tgz", 505 | "integrity": "sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==", 506 | "engines": { 507 | "node": ">= 0.6" 508 | } 509 | }, 510 | "node_modules/cookie-signature": { 511 | "version": "1.0.6", 512 | "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", 513 | "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==" 514 | }, 515 | "node_modules/cors": { 516 | "version": "2.8.5", 517 | "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", 518 | "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", 519 | "dependencies": { 520 | "object-assign": "^4", 521 | "vary": "^1" 522 | }, 523 | "engines": { 524 | "node": ">= 0.10" 525 | } 526 | }, 527 | "node_modules/debug": { 528 | "version": "2.6.9", 529 | "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", 530 | "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", 531 | "dependencies": { 532 | "ms": "2.0.0" 533 | } 534 | }, 535 | "node_modules/delayed-stream": { 536 | "version": "1.0.0", 537 | "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", 538 | "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", 539 | "license": "MIT", 540 | "engines": { 541 | "node": ">=0.4.0" 542 | } 543 | }, 544 | "node_modules/delegates": { 545 | "version": "1.0.0", 546 | "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz", 547 | "integrity": "sha512-bd2L678uiWATM6m5Z1VzNCErI3jiGzt6HGY8OVICs40JQq/HALfbyNJmp0UDakEY4pMMaN0Ly5om/B1VI/+xfQ==", 548 | "license": "MIT" 549 | }, 550 | "node_modules/depd": { 551 | "version": "2.0.0", 552 | "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", 553 | "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", 554 | "engines": { 555 | "node": ">= 0.8" 556 | } 557 | }, 558 | "node_modules/destroy": { 559 | "version": "1.2.0", 560 | "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", 561 | "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", 562 | "engines": { 563 | "node": ">= 0.8", 564 | "npm": "1.2.8000 || >= 1.4.16" 565 | } 566 | }, 567 | "node_modules/detect-libc": { 568 | "version": "2.0.3", 569 | "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.3.tgz", 570 | "integrity": "sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==", 571 | "license": "Apache-2.0", 572 | "engines": { 573 | "node": ">=8" 574 | } 575 | }, 576 | "node_modules/doctypes": { 577 | "version": "1.1.0", 578 | "resolved": "https://registry.npmjs.org/doctypes/-/doctypes-1.1.0.tgz", 579 | "integrity": "sha512-LLBi6pEqS6Do3EKQ3J0NqHWV5hhb78Pi8vvESYwyOy2c31ZEZVdtitdzsQsKb7878PEERhzUk0ftqGhG6Mz+pQ==" 580 | }, 581 | "node_modules/dotenv": { 582 | "version": "16.4.7", 583 | "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz", 584 | "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==", 585 | "engines": { 586 | "node": ">=12" 587 | }, 588 | "funding": { 589 | "url": "https://dotenvx.com" 590 | } 591 | }, 592 | "node_modules/dunder-proto": { 593 | "version": "1.0.1", 594 | "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", 595 | "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", 596 | "dependencies": { 597 | "call-bind-apply-helpers": "^1.0.1", 598 | "es-errors": "^1.3.0", 599 | "gopd": "^1.2.0" 600 | }, 601 | "engines": { 602 | "node": ">= 0.4" 603 | } 604 | }, 605 | "node_modules/ecdsa-sig-formatter": { 606 | "version": "1.0.11", 607 | "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", 608 | "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", 609 | "license": "Apache-2.0", 610 | "dependencies": { 611 | "safe-buffer": "^5.0.1" 612 | } 613 | }, 614 | "node_modules/ee-first": { 615 | "version": "1.1.1", 616 | "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", 617 | "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" 618 | }, 619 | "node_modules/emoji-regex": { 620 | "version": "8.0.0", 621 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 622 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 623 | "license": "MIT" 624 | }, 625 | "node_modules/encodeurl": { 626 | "version": "2.0.0", 627 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", 628 | "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", 629 | "engines": { 630 | "node": ">= 0.8" 631 | } 632 | }, 633 | "node_modules/es-define-property": { 634 | "version": "1.0.1", 635 | "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", 636 | "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", 637 | "engines": { 638 | "node": ">= 0.4" 639 | } 640 | }, 641 | "node_modules/es-errors": { 642 | "version": "1.3.0", 643 | "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", 644 | "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", 645 | "engines": { 646 | "node": ">= 0.4" 647 | } 648 | }, 649 | "node_modules/es-object-atoms": { 650 | "version": "1.1.1", 651 | "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", 652 | "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", 653 | "dependencies": { 654 | "es-errors": "^1.3.0" 655 | }, 656 | "engines": { 657 | "node": ">= 0.4" 658 | } 659 | }, 660 | "node_modules/es-set-tostringtag": { 661 | "version": "2.1.0", 662 | "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", 663 | "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", 664 | "license": "MIT", 665 | "dependencies": { 666 | "es-errors": "^1.3.0", 667 | "get-intrinsic": "^1.2.6", 668 | "has-tostringtag": "^1.0.2", 669 | "hasown": "^2.0.2" 670 | }, 671 | "engines": { 672 | "node": ">= 0.4" 673 | } 674 | }, 675 | "node_modules/escape-html": { 676 | "version": "1.0.3", 677 | "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", 678 | "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" 679 | }, 680 | "node_modules/etag": { 681 | "version": "1.8.1", 682 | "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", 683 | "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", 684 | "engines": { 685 | "node": ">= 0.6" 686 | } 687 | }, 688 | "node_modules/express": { 689 | "version": "4.21.2", 690 | "resolved": "https://registry.npmjs.org/express/-/express-4.21.2.tgz", 691 | "integrity": "sha512-28HqgMZAmih1Czt9ny7qr6ek2qddF4FclbMzwhCREB6OFfH+rXAnuNCwo1/wFvrtbgsQDb4kSbX9de9lFbrXnA==", 692 | "dependencies": { 693 | "accepts": "~1.3.8", 694 | "array-flatten": "1.1.1", 695 | "body-parser": "1.20.3", 696 | "content-disposition": "0.5.4", 697 | "content-type": "~1.0.4", 698 | "cookie": "0.7.1", 699 | "cookie-signature": "1.0.6", 700 | "debug": "2.6.9", 701 | "depd": "2.0.0", 702 | "encodeurl": "~2.0.0", 703 | "escape-html": "~1.0.3", 704 | "etag": "~1.8.1", 705 | "finalhandler": "1.3.1", 706 | "fresh": "0.5.2", 707 | "http-errors": "2.0.0", 708 | "merge-descriptors": "1.0.3", 709 | "methods": "~1.1.2", 710 | "on-finished": "2.4.1", 711 | "parseurl": "~1.3.3", 712 | "path-to-regexp": "0.1.12", 713 | "proxy-addr": "~2.0.7", 714 | "qs": "6.13.0", 715 | "range-parser": "~1.2.1", 716 | "safe-buffer": "5.2.1", 717 | "send": "0.19.0", 718 | "serve-static": "1.16.2", 719 | "setprototypeof": "1.2.0", 720 | "statuses": "2.0.1", 721 | "type-is": "~1.6.18", 722 | "utils-merge": "1.0.1", 723 | "vary": "~1.1.2" 724 | }, 725 | "engines": { 726 | "node": ">= 0.10.0" 727 | }, 728 | "funding": { 729 | "type": "opencollective", 730 | "url": "https://opencollective.com/express" 731 | } 732 | }, 733 | "node_modules/fill-range": { 734 | "version": "7.1.1", 735 | "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", 736 | "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", 737 | "dev": true, 738 | "dependencies": { 739 | "to-regex-range": "^5.0.1" 740 | }, 741 | "engines": { 742 | "node": ">=8" 743 | } 744 | }, 745 | "node_modules/finalhandler": { 746 | "version": "1.3.1", 747 | "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.1.tgz", 748 | "integrity": "sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ==", 749 | "dependencies": { 750 | "debug": "2.6.9", 751 | "encodeurl": "~2.0.0", 752 | "escape-html": "~1.0.3", 753 | "on-finished": "2.4.1", 754 | "parseurl": "~1.3.3", 755 | "statuses": "2.0.1", 756 | "unpipe": "~1.0.0" 757 | }, 758 | "engines": { 759 | "node": ">= 0.8" 760 | } 761 | }, 762 | "node_modules/follow-redirects": { 763 | "version": "1.15.9", 764 | "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz", 765 | "integrity": "sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==", 766 | "funding": [ 767 | { 768 | "type": "individual", 769 | "url": "https://github.com/sponsors/RubenVerborgh" 770 | } 771 | ], 772 | "license": "MIT", 773 | "engines": { 774 | "node": ">=4.0" 775 | }, 776 | "peerDependenciesMeta": { 777 | "debug": { 778 | "optional": true 779 | } 780 | } 781 | }, 782 | "node_modules/form-data": { 783 | "version": "4.0.2", 784 | "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.2.tgz", 785 | "integrity": "sha512-hGfm/slu0ZabnNt4oaRZ6uREyfCj6P4fT/n6A1rGV+Z0VdGXjfOhVUpkn6qVQONHGIFwmveGXyDs75+nr6FM8w==", 786 | "license": "MIT", 787 | "dependencies": { 788 | "asynckit": "^0.4.0", 789 | "combined-stream": "^1.0.8", 790 | "es-set-tostringtag": "^2.1.0", 791 | "mime-types": "^2.1.12" 792 | }, 793 | "engines": { 794 | "node": ">= 6" 795 | } 796 | }, 797 | "node_modules/forwarded": { 798 | "version": "0.2.0", 799 | "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", 800 | "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", 801 | "engines": { 802 | "node": ">= 0.6" 803 | } 804 | }, 805 | "node_modules/fresh": { 806 | "version": "0.5.2", 807 | "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", 808 | "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", 809 | "engines": { 810 | "node": ">= 0.6" 811 | } 812 | }, 813 | "node_modules/fs-minipass": { 814 | "version": "2.1.0", 815 | "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-2.1.0.tgz", 816 | "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", 817 | "license": "ISC", 818 | "dependencies": { 819 | "minipass": "^3.0.0" 820 | }, 821 | "engines": { 822 | "node": ">= 8" 823 | } 824 | }, 825 | "node_modules/fs-minipass/node_modules/minipass": { 826 | "version": "3.3.6", 827 | "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.3.6.tgz", 828 | "integrity": "sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw==", 829 | "license": "ISC", 830 | "dependencies": { 831 | "yallist": "^4.0.0" 832 | }, 833 | "engines": { 834 | "node": ">=8" 835 | } 836 | }, 837 | "node_modules/fs.realpath": { 838 | "version": "1.0.0", 839 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 840 | "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", 841 | "license": "ISC" 842 | }, 843 | "node_modules/fsevents": { 844 | "version": "2.3.3", 845 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", 846 | "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", 847 | "dev": true, 848 | "hasInstallScript": true, 849 | "optional": true, 850 | "os": [ 851 | "darwin" 852 | ], 853 | "engines": { 854 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 855 | } 856 | }, 857 | "node_modules/function-bind": { 858 | "version": "1.1.2", 859 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", 860 | "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", 861 | "funding": { 862 | "url": "https://github.com/sponsors/ljharb" 863 | } 864 | }, 865 | "node_modules/gauge": { 866 | "version": "3.0.2", 867 | "resolved": "https://registry.npmjs.org/gauge/-/gauge-3.0.2.tgz", 868 | "integrity": "sha512-+5J6MS/5XksCuXq++uFRsnUd7Ovu1XenbeuIuNRJxYWjgQbPuFhT14lAvsWfqfAmnwluf1OwMjz39HjfLPci0Q==", 869 | "deprecated": "This package is no longer supported.", 870 | "license": "ISC", 871 | "dependencies": { 872 | "aproba": "^1.0.3 || ^2.0.0", 873 | "color-support": "^1.1.2", 874 | "console-control-strings": "^1.0.0", 875 | "has-unicode": "^2.0.1", 876 | "object-assign": "^4.1.1", 877 | "signal-exit": "^3.0.0", 878 | "string-width": "^4.2.3", 879 | "strip-ansi": "^6.0.1", 880 | "wide-align": "^1.1.2" 881 | }, 882 | "engines": { 883 | "node": ">=10" 884 | } 885 | }, 886 | "node_modules/get-intrinsic": { 887 | "version": "1.3.0", 888 | "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", 889 | "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", 890 | "dependencies": { 891 | "call-bind-apply-helpers": "^1.0.2", 892 | "es-define-property": "^1.0.1", 893 | "es-errors": "^1.3.0", 894 | "es-object-atoms": "^1.1.1", 895 | "function-bind": "^1.1.2", 896 | "get-proto": "^1.0.1", 897 | "gopd": "^1.2.0", 898 | "has-symbols": "^1.1.0", 899 | "hasown": "^2.0.2", 900 | "math-intrinsics": "^1.1.0" 901 | }, 902 | "engines": { 903 | "node": ">= 0.4" 904 | }, 905 | "funding": { 906 | "url": "https://github.com/sponsors/ljharb" 907 | } 908 | }, 909 | "node_modules/get-proto": { 910 | "version": "1.0.1", 911 | "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", 912 | "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", 913 | "dependencies": { 914 | "dunder-proto": "^1.0.1", 915 | "es-object-atoms": "^1.0.0" 916 | }, 917 | "engines": { 918 | "node": ">= 0.4" 919 | } 920 | }, 921 | "node_modules/glob": { 922 | "version": "7.2.3", 923 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", 924 | "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", 925 | "deprecated": "Glob versions prior to v9 are no longer supported", 926 | "license": "ISC", 927 | "dependencies": { 928 | "fs.realpath": "^1.0.0", 929 | "inflight": "^1.0.4", 930 | "inherits": "2", 931 | "minimatch": "^3.1.1", 932 | "once": "^1.3.0", 933 | "path-is-absolute": "^1.0.0" 934 | }, 935 | "engines": { 936 | "node": "*" 937 | }, 938 | "funding": { 939 | "url": "https://github.com/sponsors/isaacs" 940 | } 941 | }, 942 | "node_modules/glob-parent": { 943 | "version": "5.1.2", 944 | "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", 945 | "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", 946 | "dev": true, 947 | "dependencies": { 948 | "is-glob": "^4.0.1" 949 | }, 950 | "engines": { 951 | "node": ">= 6" 952 | } 953 | }, 954 | "node_modules/gopd": { 955 | "version": "1.2.0", 956 | "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", 957 | "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", 958 | "engines": { 959 | "node": ">= 0.4" 960 | }, 961 | "funding": { 962 | "url": "https://github.com/sponsors/ljharb" 963 | } 964 | }, 965 | "node_modules/has-flag": { 966 | "version": "3.0.0", 967 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 968 | "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", 969 | "dev": true, 970 | "engines": { 971 | "node": ">=4" 972 | } 973 | }, 974 | "node_modules/has-symbols": { 975 | "version": "1.1.0", 976 | "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", 977 | "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", 978 | "engines": { 979 | "node": ">= 0.4" 980 | }, 981 | "funding": { 982 | "url": "https://github.com/sponsors/ljharb" 983 | } 984 | }, 985 | "node_modules/has-tostringtag": { 986 | "version": "1.0.2", 987 | "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", 988 | "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", 989 | "dependencies": { 990 | "has-symbols": "^1.0.3" 991 | }, 992 | "engines": { 993 | "node": ">= 0.4" 994 | }, 995 | "funding": { 996 | "url": "https://github.com/sponsors/ljharb" 997 | } 998 | }, 999 | "node_modules/has-unicode": { 1000 | "version": "2.0.1", 1001 | "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz", 1002 | "integrity": "sha512-8Rf9Y83NBReMnx0gFzA8JImQACstCYWUplepDa9xprwwtmgEZUF0h/i5xSA625zB/I37EtrswSST6OXxwaaIJQ==", 1003 | "license": "ISC" 1004 | }, 1005 | "node_modules/hasown": { 1006 | "version": "2.0.2", 1007 | "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", 1008 | "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", 1009 | "dependencies": { 1010 | "function-bind": "^1.1.2" 1011 | }, 1012 | "engines": { 1013 | "node": ">= 0.4" 1014 | } 1015 | }, 1016 | "node_modules/helmet": { 1017 | "version": "8.0.0", 1018 | "resolved": "https://registry.npmjs.org/helmet/-/helmet-8.0.0.tgz", 1019 | "integrity": "sha512-VyusHLEIIO5mjQPUI1wpOAEu+wl6Q0998jzTxqUYGE45xCIcAxy3MsbEK/yyJUJ3ADeMoB6MornPH6GMWAf+Pw==", 1020 | "engines": { 1021 | "node": ">=18.0.0" 1022 | } 1023 | }, 1024 | "node_modules/http-errors": { 1025 | "version": "2.0.0", 1026 | "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", 1027 | "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", 1028 | "dependencies": { 1029 | "depd": "2.0.0", 1030 | "inherits": "2.0.4", 1031 | "setprototypeof": "1.2.0", 1032 | "statuses": "2.0.1", 1033 | "toidentifier": "1.0.1" 1034 | }, 1035 | "engines": { 1036 | "node": ">= 0.8" 1037 | } 1038 | }, 1039 | "node_modules/https-proxy-agent": { 1040 | "version": "5.0.1", 1041 | "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz", 1042 | "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==", 1043 | "license": "MIT", 1044 | "dependencies": { 1045 | "agent-base": "6", 1046 | "debug": "4" 1047 | }, 1048 | "engines": { 1049 | "node": ">= 6" 1050 | } 1051 | }, 1052 | "node_modules/https-proxy-agent/node_modules/debug": { 1053 | "version": "4.4.0", 1054 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", 1055 | "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", 1056 | "license": "MIT", 1057 | "dependencies": { 1058 | "ms": "^2.1.3" 1059 | }, 1060 | "engines": { 1061 | "node": ">=6.0" 1062 | }, 1063 | "peerDependenciesMeta": { 1064 | "supports-color": { 1065 | "optional": true 1066 | } 1067 | } 1068 | }, 1069 | "node_modules/https-proxy-agent/node_modules/ms": { 1070 | "version": "2.1.3", 1071 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1072 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 1073 | "license": "MIT" 1074 | }, 1075 | "node_modules/iconv-lite": { 1076 | "version": "0.4.24", 1077 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 1078 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 1079 | "dependencies": { 1080 | "safer-buffer": ">= 2.1.2 < 3" 1081 | }, 1082 | "engines": { 1083 | "node": ">=0.10.0" 1084 | } 1085 | }, 1086 | "node_modules/ignore-by-default": { 1087 | "version": "1.0.1", 1088 | "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", 1089 | "integrity": "sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==", 1090 | "dev": true 1091 | }, 1092 | "node_modules/inflight": { 1093 | "version": "1.0.6", 1094 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 1095 | "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", 1096 | "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", 1097 | "license": "ISC", 1098 | "dependencies": { 1099 | "once": "^1.3.0", 1100 | "wrappy": "1" 1101 | } 1102 | }, 1103 | "node_modules/inherits": { 1104 | "version": "2.0.4", 1105 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 1106 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" 1107 | }, 1108 | "node_modules/ipaddr.js": { 1109 | "version": "1.9.1", 1110 | "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", 1111 | "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", 1112 | "engines": { 1113 | "node": ">= 0.10" 1114 | } 1115 | }, 1116 | "node_modules/is-binary-path": { 1117 | "version": "2.1.0", 1118 | "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", 1119 | "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", 1120 | "dev": true, 1121 | "dependencies": { 1122 | "binary-extensions": "^2.0.0" 1123 | }, 1124 | "engines": { 1125 | "node": ">=8" 1126 | } 1127 | }, 1128 | "node_modules/is-core-module": { 1129 | "version": "2.16.1", 1130 | "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", 1131 | "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", 1132 | "dependencies": { 1133 | "hasown": "^2.0.2" 1134 | }, 1135 | "engines": { 1136 | "node": ">= 0.4" 1137 | }, 1138 | "funding": { 1139 | "url": "https://github.com/sponsors/ljharb" 1140 | } 1141 | }, 1142 | "node_modules/is-expression": { 1143 | "version": "4.0.0", 1144 | "resolved": "https://registry.npmjs.org/is-expression/-/is-expression-4.0.0.tgz", 1145 | "integrity": "sha512-zMIXX63sxzG3XrkHkrAPvm/OVZVSCPNkwMHU8oTX7/U3AL78I0QXCEICXUM13BIa8TYGZ68PiTKfQz3yaTNr4A==", 1146 | "dependencies": { 1147 | "acorn": "^7.1.1", 1148 | "object-assign": "^4.1.1" 1149 | } 1150 | }, 1151 | "node_modules/is-extglob": { 1152 | "version": "2.1.1", 1153 | "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", 1154 | "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", 1155 | "dev": true, 1156 | "engines": { 1157 | "node": ">=0.10.0" 1158 | } 1159 | }, 1160 | "node_modules/is-fullwidth-code-point": { 1161 | "version": "3.0.0", 1162 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 1163 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 1164 | "license": "MIT", 1165 | "engines": { 1166 | "node": ">=8" 1167 | } 1168 | }, 1169 | "node_modules/is-glob": { 1170 | "version": "4.0.3", 1171 | "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", 1172 | "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", 1173 | "dev": true, 1174 | "dependencies": { 1175 | "is-extglob": "^2.1.1" 1176 | }, 1177 | "engines": { 1178 | "node": ">=0.10.0" 1179 | } 1180 | }, 1181 | "node_modules/is-number": { 1182 | "version": "7.0.0", 1183 | "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", 1184 | "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", 1185 | "dev": true, 1186 | "engines": { 1187 | "node": ">=0.12.0" 1188 | } 1189 | }, 1190 | "node_modules/is-promise": { 1191 | "version": "2.2.2", 1192 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.2.2.tgz", 1193 | "integrity": "sha512-+lP4/6lKUBfQjZ2pdxThZvLUAafmZb8OAxFb8XXtiQmS35INgr85hdOGoEs124ez1FCnZJt6jau/T+alh58QFQ==" 1194 | }, 1195 | "node_modules/is-regex": { 1196 | "version": "1.2.1", 1197 | "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.2.1.tgz", 1198 | "integrity": "sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==", 1199 | "dependencies": { 1200 | "call-bound": "^1.0.2", 1201 | "gopd": "^1.2.0", 1202 | "has-tostringtag": "^1.0.2", 1203 | "hasown": "^2.0.2" 1204 | }, 1205 | "engines": { 1206 | "node": ">= 0.4" 1207 | }, 1208 | "funding": { 1209 | "url": "https://github.com/sponsors/ljharb" 1210 | } 1211 | }, 1212 | "node_modules/js-stringify": { 1213 | "version": "1.0.2", 1214 | "resolved": "https://registry.npmjs.org/js-stringify/-/js-stringify-1.0.2.tgz", 1215 | "integrity": "sha512-rtS5ATOo2Q5k1G+DADISilDA6lv79zIiwFd6CcjuIxGKLFm5C+RLImRscVap9k55i+MOZwgliw+NejvkLuGD5g==" 1216 | }, 1217 | "node_modules/jsonwebtoken": { 1218 | "version": "9.0.2", 1219 | "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz", 1220 | "integrity": "sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==", 1221 | "license": "MIT", 1222 | "dependencies": { 1223 | "jws": "^3.2.2", 1224 | "lodash.includes": "^4.3.0", 1225 | "lodash.isboolean": "^3.0.3", 1226 | "lodash.isinteger": "^4.0.4", 1227 | "lodash.isnumber": "^3.0.3", 1228 | "lodash.isplainobject": "^4.0.6", 1229 | "lodash.isstring": "^4.0.1", 1230 | "lodash.once": "^4.0.0", 1231 | "ms": "^2.1.1", 1232 | "semver": "^7.5.4" 1233 | }, 1234 | "engines": { 1235 | "node": ">=12", 1236 | "npm": ">=6" 1237 | } 1238 | }, 1239 | "node_modules/jsonwebtoken/node_modules/ms": { 1240 | "version": "2.1.3", 1241 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1242 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 1243 | "license": "MIT" 1244 | }, 1245 | "node_modules/jstransformer": { 1246 | "version": "1.0.0", 1247 | "resolved": "https://registry.npmjs.org/jstransformer/-/jstransformer-1.0.0.tgz", 1248 | "integrity": "sha512-C9YK3Rf8q6VAPDCCU9fnqo3mAfOH6vUGnMcP4AQAYIEpWtfGLpwOTmZ+igtdK5y+VvI2n3CyYSzy4Qh34eq24A==", 1249 | "dependencies": { 1250 | "is-promise": "^2.0.0", 1251 | "promise": "^7.0.1" 1252 | } 1253 | }, 1254 | "node_modules/jwa": { 1255 | "version": "1.4.1", 1256 | "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", 1257 | "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", 1258 | "license": "MIT", 1259 | "dependencies": { 1260 | "buffer-equal-constant-time": "1.0.1", 1261 | "ecdsa-sig-formatter": "1.0.11", 1262 | "safe-buffer": "^5.0.1" 1263 | } 1264 | }, 1265 | "node_modules/jws": { 1266 | "version": "3.2.2", 1267 | "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", 1268 | "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", 1269 | "license": "MIT", 1270 | "dependencies": { 1271 | "jwa": "^1.4.1", 1272 | "safe-buffer": "^5.0.1" 1273 | } 1274 | }, 1275 | "node_modules/kareem": { 1276 | "version": "2.6.3", 1277 | "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.6.3.tgz", 1278 | "integrity": "sha512-C3iHfuGUXK2u8/ipq9LfjFfXFxAZMQJJq7vLS45r3D9Y2xQ/m4S8zaR4zMLFWh9AsNPXmcFfUDhTEO8UIC/V6Q==", 1279 | "engines": { 1280 | "node": ">=12.0.0" 1281 | } 1282 | }, 1283 | "node_modules/lodash.includes": { 1284 | "version": "4.3.0", 1285 | "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", 1286 | "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==", 1287 | "license": "MIT" 1288 | }, 1289 | "node_modules/lodash.isboolean": { 1290 | "version": "3.0.3", 1291 | "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", 1292 | "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==", 1293 | "license": "MIT" 1294 | }, 1295 | "node_modules/lodash.isinteger": { 1296 | "version": "4.0.4", 1297 | "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", 1298 | "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==", 1299 | "license": "MIT" 1300 | }, 1301 | "node_modules/lodash.isnumber": { 1302 | "version": "3.0.3", 1303 | "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", 1304 | "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==", 1305 | "license": "MIT" 1306 | }, 1307 | "node_modules/lodash.isplainobject": { 1308 | "version": "4.0.6", 1309 | "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", 1310 | "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==", 1311 | "license": "MIT" 1312 | }, 1313 | "node_modules/lodash.isstring": { 1314 | "version": "4.0.1", 1315 | "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", 1316 | "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==", 1317 | "license": "MIT" 1318 | }, 1319 | "node_modules/lodash.once": { 1320 | "version": "4.1.1", 1321 | "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", 1322 | "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==", 1323 | "license": "MIT" 1324 | }, 1325 | "node_modules/make-dir": { 1326 | "version": "3.1.0", 1327 | "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", 1328 | "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", 1329 | "license": "MIT", 1330 | "dependencies": { 1331 | "semver": "^6.0.0" 1332 | }, 1333 | "engines": { 1334 | "node": ">=8" 1335 | }, 1336 | "funding": { 1337 | "url": "https://github.com/sponsors/sindresorhus" 1338 | } 1339 | }, 1340 | "node_modules/make-dir/node_modules/semver": { 1341 | "version": "6.3.1", 1342 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", 1343 | "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", 1344 | "license": "ISC", 1345 | "bin": { 1346 | "semver": "bin/semver.js" 1347 | } 1348 | }, 1349 | "node_modules/math-intrinsics": { 1350 | "version": "1.1.0", 1351 | "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", 1352 | "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", 1353 | "engines": { 1354 | "node": ">= 0.4" 1355 | } 1356 | }, 1357 | "node_modules/media-typer": { 1358 | "version": "0.3.0", 1359 | "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", 1360 | "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", 1361 | "engines": { 1362 | "node": ">= 0.6" 1363 | } 1364 | }, 1365 | "node_modules/memory-pager": { 1366 | "version": "1.5.0", 1367 | "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", 1368 | "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==" 1369 | }, 1370 | "node_modules/merge-descriptors": { 1371 | "version": "1.0.3", 1372 | "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", 1373 | "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", 1374 | "funding": { 1375 | "url": "https://github.com/sponsors/sindresorhus" 1376 | } 1377 | }, 1378 | "node_modules/methods": { 1379 | "version": "1.1.2", 1380 | "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", 1381 | "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", 1382 | "engines": { 1383 | "node": ">= 0.6" 1384 | } 1385 | }, 1386 | "node_modules/mime": { 1387 | "version": "1.6.0", 1388 | "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", 1389 | "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", 1390 | "bin": { 1391 | "mime": "cli.js" 1392 | }, 1393 | "engines": { 1394 | "node": ">=4" 1395 | } 1396 | }, 1397 | "node_modules/mime-db": { 1398 | "version": "1.52.0", 1399 | "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", 1400 | "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", 1401 | "engines": { 1402 | "node": ">= 0.6" 1403 | } 1404 | }, 1405 | "node_modules/mime-types": { 1406 | "version": "2.1.35", 1407 | "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", 1408 | "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", 1409 | "dependencies": { 1410 | "mime-db": "1.52.0" 1411 | }, 1412 | "engines": { 1413 | "node": ">= 0.6" 1414 | } 1415 | }, 1416 | "node_modules/minimatch": { 1417 | "version": "3.1.2", 1418 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 1419 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 1420 | "dependencies": { 1421 | "brace-expansion": "^1.1.7" 1422 | }, 1423 | "engines": { 1424 | "node": "*" 1425 | } 1426 | }, 1427 | "node_modules/minipass": { 1428 | "version": "5.0.0", 1429 | "resolved": "https://registry.npmjs.org/minipass/-/minipass-5.0.0.tgz", 1430 | "integrity": "sha512-3FnjYuehv9k6ovOEbyOswadCDPX1piCfhV8ncmYtHOjuPwylVWsghTLo7rabjC3Rx5xD4HDx8Wm1xnMF7S5qFQ==", 1431 | "license": "ISC", 1432 | "engines": { 1433 | "node": ">=8" 1434 | } 1435 | }, 1436 | "node_modules/minizlib": { 1437 | "version": "2.1.2", 1438 | "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-2.1.2.tgz", 1439 | "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", 1440 | "license": "MIT", 1441 | "dependencies": { 1442 | "minipass": "^3.0.0", 1443 | "yallist": "^4.0.0" 1444 | }, 1445 | "engines": { 1446 | "node": ">= 8" 1447 | } 1448 | }, 1449 | "node_modules/minizlib/node_modules/minipass": { 1450 | "version": "3.3.6", 1451 | "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.3.6.tgz", 1452 | "integrity": "sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw==", 1453 | "license": "ISC", 1454 | "dependencies": { 1455 | "yallist": "^4.0.0" 1456 | }, 1457 | "engines": { 1458 | "node": ">=8" 1459 | } 1460 | }, 1461 | "node_modules/mkdirp": { 1462 | "version": "1.0.4", 1463 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", 1464 | "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", 1465 | "license": "MIT", 1466 | "bin": { 1467 | "mkdirp": "bin/cmd.js" 1468 | }, 1469 | "engines": { 1470 | "node": ">=10" 1471 | } 1472 | }, 1473 | "node_modules/mongodb": { 1474 | "version": "6.14.2", 1475 | "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-6.14.2.tgz", 1476 | "integrity": "sha512-kMEHNo0F3P6QKDq17zcDuPeaywK/YaJVCEQRzPF3TOM/Bl9MFg64YE5Tu7ifj37qZJMhwU1tl2Ioivws5gRG5Q==", 1477 | "dependencies": { 1478 | "@mongodb-js/saslprep": "^1.1.9", 1479 | "bson": "^6.10.3", 1480 | "mongodb-connection-string-url": "^3.0.0" 1481 | }, 1482 | "engines": { 1483 | "node": ">=16.20.1" 1484 | }, 1485 | "peerDependencies": { 1486 | "@aws-sdk/credential-providers": "^3.188.0", 1487 | "@mongodb-js/zstd": "^1.1.0 || ^2.0.0", 1488 | "gcp-metadata": "^5.2.0", 1489 | "kerberos": "^2.0.1", 1490 | "mongodb-client-encryption": ">=6.0.0 <7", 1491 | "snappy": "^7.2.2", 1492 | "socks": "^2.7.1" 1493 | }, 1494 | "peerDependenciesMeta": { 1495 | "@aws-sdk/credential-providers": { 1496 | "optional": true 1497 | }, 1498 | "@mongodb-js/zstd": { 1499 | "optional": true 1500 | }, 1501 | "gcp-metadata": { 1502 | "optional": true 1503 | }, 1504 | "kerberos": { 1505 | "optional": true 1506 | }, 1507 | "mongodb-client-encryption": { 1508 | "optional": true 1509 | }, 1510 | "snappy": { 1511 | "optional": true 1512 | }, 1513 | "socks": { 1514 | "optional": true 1515 | } 1516 | } 1517 | }, 1518 | "node_modules/mongodb-connection-string-url": { 1519 | "version": "3.0.2", 1520 | "resolved": "https://registry.npmjs.org/mongodb-connection-string-url/-/mongodb-connection-string-url-3.0.2.tgz", 1521 | "integrity": "sha512-rMO7CGo/9BFwyZABcKAWL8UJwH/Kc2x0g72uhDWzG48URRax5TCIcJ7Rc3RZqffZzO/Gwff/jyKwCU9TN8gehA==", 1522 | "dependencies": { 1523 | "@types/whatwg-url": "^11.0.2", 1524 | "whatwg-url": "^14.1.0 || ^13.0.0" 1525 | } 1526 | }, 1527 | "node_modules/mongoose": { 1528 | "version": "8.12.1", 1529 | "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-8.12.1.tgz", 1530 | "integrity": "sha512-UW22y8QFVYmrb36hm8cGncfn4ARc/XsYWQwRTaj0gxtQk1rDuhzDO1eBantS+hTTatfAIS96LlRCJrcNHvW5+Q==", 1531 | "dependencies": { 1532 | "bson": "^6.10.3", 1533 | "kareem": "2.6.3", 1534 | "mongodb": "~6.14.0", 1535 | "mpath": "0.9.0", 1536 | "mquery": "5.0.0", 1537 | "ms": "2.1.3", 1538 | "sift": "17.1.3" 1539 | }, 1540 | "engines": { 1541 | "node": ">=16.20.1" 1542 | }, 1543 | "funding": { 1544 | "type": "opencollective", 1545 | "url": "https://opencollective.com/mongoose" 1546 | } 1547 | }, 1548 | "node_modules/mongoose/node_modules/ms": { 1549 | "version": "2.1.3", 1550 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1551 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 1552 | }, 1553 | "node_modules/morgan": { 1554 | "version": "1.10.0", 1555 | "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.10.0.tgz", 1556 | "integrity": "sha512-AbegBVI4sh6El+1gNwvD5YIck7nSA36weD7xvIxG4in80j/UoK8AEGaWnnz8v1GxonMCltmlNs5ZKbGvl9b1XQ==", 1557 | "dev": true, 1558 | "dependencies": { 1559 | "basic-auth": "~2.0.1", 1560 | "debug": "2.6.9", 1561 | "depd": "~2.0.0", 1562 | "on-finished": "~2.3.0", 1563 | "on-headers": "~1.0.2" 1564 | }, 1565 | "engines": { 1566 | "node": ">= 0.8.0" 1567 | } 1568 | }, 1569 | "node_modules/morgan/node_modules/on-finished": { 1570 | "version": "2.3.0", 1571 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", 1572 | "integrity": "sha512-ikqdkGAAyf/X/gPhXGvfgAytDZtDbr+bkNUJ0N9h5MI/dmdgCs3l6hoHrcUv41sRKew3jIwrp4qQDXiK99Utww==", 1573 | "dev": true, 1574 | "dependencies": { 1575 | "ee-first": "1.1.1" 1576 | }, 1577 | "engines": { 1578 | "node": ">= 0.8" 1579 | } 1580 | }, 1581 | "node_modules/mpath": { 1582 | "version": "0.9.0", 1583 | "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.9.0.tgz", 1584 | "integrity": "sha512-ikJRQTk8hw5DEoFVxHG1Gn9T/xcjtdnOKIU1JTmGjZZlg9LST2mBLmcX3/ICIbgJydT2GOc15RnNy5mHmzfSew==", 1585 | "engines": { 1586 | "node": ">=4.0.0" 1587 | } 1588 | }, 1589 | "node_modules/mquery": { 1590 | "version": "5.0.0", 1591 | "resolved": "https://registry.npmjs.org/mquery/-/mquery-5.0.0.tgz", 1592 | "integrity": "sha512-iQMncpmEK8R8ncT8HJGsGc9Dsp8xcgYMVSbs5jgnm1lFHTZqMJTUWTDx1LBO8+mK3tPNZWFLBghQEIOULSTHZg==", 1593 | "dependencies": { 1594 | "debug": "4.x" 1595 | }, 1596 | "engines": { 1597 | "node": ">=14.0.0" 1598 | } 1599 | }, 1600 | "node_modules/mquery/node_modules/debug": { 1601 | "version": "4.4.0", 1602 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", 1603 | "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", 1604 | "dependencies": { 1605 | "ms": "^2.1.3" 1606 | }, 1607 | "engines": { 1608 | "node": ">=6.0" 1609 | }, 1610 | "peerDependenciesMeta": { 1611 | "supports-color": { 1612 | "optional": true 1613 | } 1614 | } 1615 | }, 1616 | "node_modules/mquery/node_modules/ms": { 1617 | "version": "2.1.3", 1618 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1619 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 1620 | }, 1621 | "node_modules/ms": { 1622 | "version": "2.0.0", 1623 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", 1624 | "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" 1625 | }, 1626 | "node_modules/negotiator": { 1627 | "version": "0.6.3", 1628 | "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", 1629 | "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", 1630 | "engines": { 1631 | "node": ">= 0.6" 1632 | } 1633 | }, 1634 | "node_modules/node-addon-api": { 1635 | "version": "5.1.0", 1636 | "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-5.1.0.tgz", 1637 | "integrity": "sha512-eh0GgfEkpnoWDq+VY8OyvYhFEzBk6jIYbRKdIlyTiAXIVJ8PyBaKb0rp7oDtoddbdoHWhq8wwr+XZ81F1rpNdA==", 1638 | "license": "MIT" 1639 | }, 1640 | "node_modules/node-fetch": { 1641 | "version": "2.7.0", 1642 | "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", 1643 | "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", 1644 | "license": "MIT", 1645 | "dependencies": { 1646 | "whatwg-url": "^5.0.0" 1647 | }, 1648 | "engines": { 1649 | "node": "4.x || >=6.0.0" 1650 | }, 1651 | "peerDependencies": { 1652 | "encoding": "^0.1.0" 1653 | }, 1654 | "peerDependenciesMeta": { 1655 | "encoding": { 1656 | "optional": true 1657 | } 1658 | } 1659 | }, 1660 | "node_modules/node-fetch/node_modules/tr46": { 1661 | "version": "0.0.3", 1662 | "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", 1663 | "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", 1664 | "license": "MIT" 1665 | }, 1666 | "node_modules/node-fetch/node_modules/webidl-conversions": { 1667 | "version": "3.0.1", 1668 | "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", 1669 | "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", 1670 | "license": "BSD-2-Clause" 1671 | }, 1672 | "node_modules/node-fetch/node_modules/whatwg-url": { 1673 | "version": "5.0.0", 1674 | "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", 1675 | "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", 1676 | "license": "MIT", 1677 | "dependencies": { 1678 | "tr46": "~0.0.3", 1679 | "webidl-conversions": "^3.0.0" 1680 | } 1681 | }, 1682 | "node_modules/nodemon": { 1683 | "version": "3.1.9", 1684 | "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-3.1.9.tgz", 1685 | "integrity": "sha512-hdr1oIb2p6ZSxu3PB2JWWYS7ZQ0qvaZsc3hK8DR8f02kRzc8rjYmxAIvdz+aYC+8F2IjNaB7HMcSDg8nQpJxyg==", 1686 | "dev": true, 1687 | "dependencies": { 1688 | "chokidar": "^3.5.2", 1689 | "debug": "^4", 1690 | "ignore-by-default": "^1.0.1", 1691 | "minimatch": "^3.1.2", 1692 | "pstree.remy": "^1.1.8", 1693 | "semver": "^7.5.3", 1694 | "simple-update-notifier": "^2.0.0", 1695 | "supports-color": "^5.5.0", 1696 | "touch": "^3.1.0", 1697 | "undefsafe": "^2.0.5" 1698 | }, 1699 | "bin": { 1700 | "nodemon": "bin/nodemon.js" 1701 | }, 1702 | "engines": { 1703 | "node": ">=10" 1704 | }, 1705 | "funding": { 1706 | "type": "opencollective", 1707 | "url": "https://opencollective.com/nodemon" 1708 | } 1709 | }, 1710 | "node_modules/nodemon/node_modules/debug": { 1711 | "version": "4.4.0", 1712 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", 1713 | "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", 1714 | "dev": true, 1715 | "dependencies": { 1716 | "ms": "^2.1.3" 1717 | }, 1718 | "engines": { 1719 | "node": ">=6.0" 1720 | }, 1721 | "peerDependenciesMeta": { 1722 | "supports-color": { 1723 | "optional": true 1724 | } 1725 | } 1726 | }, 1727 | "node_modules/nodemon/node_modules/ms": { 1728 | "version": "2.1.3", 1729 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 1730 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", 1731 | "dev": true 1732 | }, 1733 | "node_modules/nopt": { 1734 | "version": "5.0.0", 1735 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-5.0.0.tgz", 1736 | "integrity": "sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==", 1737 | "license": "ISC", 1738 | "dependencies": { 1739 | "abbrev": "1" 1740 | }, 1741 | "bin": { 1742 | "nopt": "bin/nopt.js" 1743 | }, 1744 | "engines": { 1745 | "node": ">=6" 1746 | } 1747 | }, 1748 | "node_modules/normalize-path": { 1749 | "version": "3.0.0", 1750 | "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", 1751 | "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", 1752 | "dev": true, 1753 | "engines": { 1754 | "node": ">=0.10.0" 1755 | } 1756 | }, 1757 | "node_modules/npmlog": { 1758 | "version": "5.0.1", 1759 | "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-5.0.1.tgz", 1760 | "integrity": "sha512-AqZtDUWOMKs1G/8lwylVjrdYgqA4d9nu8hc+0gzRxlDb1I10+FHBGMXs6aiQHFdCUUlqH99MUMuLfzWDNDtfxw==", 1761 | "deprecated": "This package is no longer supported.", 1762 | "license": "ISC", 1763 | "dependencies": { 1764 | "are-we-there-yet": "^2.0.0", 1765 | "console-control-strings": "^1.1.0", 1766 | "gauge": "^3.0.0", 1767 | "set-blocking": "^2.0.0" 1768 | } 1769 | }, 1770 | "node_modules/object-assign": { 1771 | "version": "4.1.1", 1772 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 1773 | "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", 1774 | "engines": { 1775 | "node": ">=0.10.0" 1776 | } 1777 | }, 1778 | "node_modules/object-inspect": { 1779 | "version": "1.13.4", 1780 | "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", 1781 | "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", 1782 | "engines": { 1783 | "node": ">= 0.4" 1784 | }, 1785 | "funding": { 1786 | "url": "https://github.com/sponsors/ljharb" 1787 | } 1788 | }, 1789 | "node_modules/on-finished": { 1790 | "version": "2.4.1", 1791 | "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", 1792 | "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", 1793 | "dependencies": { 1794 | "ee-first": "1.1.1" 1795 | }, 1796 | "engines": { 1797 | "node": ">= 0.8" 1798 | } 1799 | }, 1800 | "node_modules/on-headers": { 1801 | "version": "1.0.2", 1802 | "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", 1803 | "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==", 1804 | "dev": true, 1805 | "engines": { 1806 | "node": ">= 0.8" 1807 | } 1808 | }, 1809 | "node_modules/once": { 1810 | "version": "1.4.0", 1811 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 1812 | "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", 1813 | "license": "ISC", 1814 | "dependencies": { 1815 | "wrappy": "1" 1816 | } 1817 | }, 1818 | "node_modules/parseurl": { 1819 | "version": "1.3.3", 1820 | "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", 1821 | "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", 1822 | "engines": { 1823 | "node": ">= 0.8" 1824 | } 1825 | }, 1826 | "node_modules/path-is-absolute": { 1827 | "version": "1.0.1", 1828 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 1829 | "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", 1830 | "license": "MIT", 1831 | "engines": { 1832 | "node": ">=0.10.0" 1833 | } 1834 | }, 1835 | "node_modules/path-parse": { 1836 | "version": "1.0.7", 1837 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", 1838 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==" 1839 | }, 1840 | "node_modules/path-to-regexp": { 1841 | "version": "0.1.12", 1842 | "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz", 1843 | "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==" 1844 | }, 1845 | "node_modules/picomatch": { 1846 | "version": "2.3.1", 1847 | "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", 1848 | "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", 1849 | "dev": true, 1850 | "engines": { 1851 | "node": ">=8.6" 1852 | }, 1853 | "funding": { 1854 | "url": "https://github.com/sponsors/jonschlinkert" 1855 | } 1856 | }, 1857 | "node_modules/promise": { 1858 | "version": "7.3.1", 1859 | "resolved": "https://registry.npmjs.org/promise/-/promise-7.3.1.tgz", 1860 | "integrity": "sha512-nolQXZ/4L+bP/UGlkfaIujX9BKxGwmQ9OT4mOt5yvy8iK1h3wqTEJCijzGANTCCl9nWjY41juyAn2K3Q1hLLTg==", 1861 | "dependencies": { 1862 | "asap": "~2.0.3" 1863 | } 1864 | }, 1865 | "node_modules/proxy-addr": { 1866 | "version": "2.0.7", 1867 | "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", 1868 | "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", 1869 | "dependencies": { 1870 | "forwarded": "0.2.0", 1871 | "ipaddr.js": "1.9.1" 1872 | }, 1873 | "engines": { 1874 | "node": ">= 0.10" 1875 | } 1876 | }, 1877 | "node_modules/proxy-from-env": { 1878 | "version": "1.1.0", 1879 | "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", 1880 | "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", 1881 | "license": "MIT" 1882 | }, 1883 | "node_modules/pstree.remy": { 1884 | "version": "1.1.8", 1885 | "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", 1886 | "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", 1887 | "dev": true 1888 | }, 1889 | "node_modules/pug": { 1890 | "version": "3.0.3", 1891 | "resolved": "https://registry.npmjs.org/pug/-/pug-3.0.3.tgz", 1892 | "integrity": "sha512-uBi6kmc9f3SZ3PXxqcHiUZLmIXgfgWooKWXcwSGwQd2Zi5Rb0bT14+8CJjJgI8AB+nndLaNgHGrcc6bPIB665g==", 1893 | "dependencies": { 1894 | "pug-code-gen": "^3.0.3", 1895 | "pug-filters": "^4.0.0", 1896 | "pug-lexer": "^5.0.1", 1897 | "pug-linker": "^4.0.0", 1898 | "pug-load": "^3.0.0", 1899 | "pug-parser": "^6.0.0", 1900 | "pug-runtime": "^3.0.1", 1901 | "pug-strip-comments": "^2.0.0" 1902 | } 1903 | }, 1904 | "node_modules/pug-attrs": { 1905 | "version": "3.0.0", 1906 | "resolved": "https://registry.npmjs.org/pug-attrs/-/pug-attrs-3.0.0.tgz", 1907 | "integrity": "sha512-azINV9dUtzPMFQktvTXciNAfAuVh/L/JCl0vtPCwvOA21uZrC08K/UnmrL+SXGEVc1FwzjW62+xw5S/uaLj6cA==", 1908 | "dependencies": { 1909 | "constantinople": "^4.0.1", 1910 | "js-stringify": "^1.0.2", 1911 | "pug-runtime": "^3.0.0" 1912 | } 1913 | }, 1914 | "node_modules/pug-code-gen": { 1915 | "version": "3.0.3", 1916 | "resolved": "https://registry.npmjs.org/pug-code-gen/-/pug-code-gen-3.0.3.tgz", 1917 | "integrity": "sha512-cYQg0JW0w32Ux+XTeZnBEeuWrAY7/HNE6TWnhiHGnnRYlCgyAUPoyh9KzCMa9WhcJlJ1AtQqpEYHc+vbCzA+Aw==", 1918 | "dependencies": { 1919 | "constantinople": "^4.0.1", 1920 | "doctypes": "^1.1.0", 1921 | "js-stringify": "^1.0.2", 1922 | "pug-attrs": "^3.0.0", 1923 | "pug-error": "^2.1.0", 1924 | "pug-runtime": "^3.0.1", 1925 | "void-elements": "^3.1.0", 1926 | "with": "^7.0.0" 1927 | } 1928 | }, 1929 | "node_modules/pug-error": { 1930 | "version": "2.1.0", 1931 | "resolved": "https://registry.npmjs.org/pug-error/-/pug-error-2.1.0.tgz", 1932 | "integrity": "sha512-lv7sU9e5Jk8IeUheHata6/UThZ7RK2jnaaNztxfPYUY+VxZyk/ePVaNZ/vwmH8WqGvDz3LrNYt/+gA55NDg6Pg==" 1933 | }, 1934 | "node_modules/pug-filters": { 1935 | "version": "4.0.0", 1936 | "resolved": "https://registry.npmjs.org/pug-filters/-/pug-filters-4.0.0.tgz", 1937 | "integrity": "sha512-yeNFtq5Yxmfz0f9z2rMXGw/8/4i1cCFecw/Q7+D0V2DdtII5UvqE12VaZ2AY7ri6o5RNXiweGH79OCq+2RQU4A==", 1938 | "dependencies": { 1939 | "constantinople": "^4.0.1", 1940 | "jstransformer": "1.0.0", 1941 | "pug-error": "^2.0.0", 1942 | "pug-walk": "^2.0.0", 1943 | "resolve": "^1.15.1" 1944 | } 1945 | }, 1946 | "node_modules/pug-lexer": { 1947 | "version": "5.0.1", 1948 | "resolved": "https://registry.npmjs.org/pug-lexer/-/pug-lexer-5.0.1.tgz", 1949 | "integrity": "sha512-0I6C62+keXlZPZkOJeVam9aBLVP2EnbeDw3An+k0/QlqdwH6rv8284nko14Na7c0TtqtogfWXcRoFE4O4Ff20w==", 1950 | "dependencies": { 1951 | "character-parser": "^2.2.0", 1952 | "is-expression": "^4.0.0", 1953 | "pug-error": "^2.0.0" 1954 | } 1955 | }, 1956 | "node_modules/pug-linker": { 1957 | "version": "4.0.0", 1958 | "resolved": "https://registry.npmjs.org/pug-linker/-/pug-linker-4.0.0.tgz", 1959 | "integrity": "sha512-gjD1yzp0yxbQqnzBAdlhbgoJL5qIFJw78juN1NpTLt/mfPJ5VgC4BvkoD3G23qKzJtIIXBbcCt6FioLSFLOHdw==", 1960 | "dependencies": { 1961 | "pug-error": "^2.0.0", 1962 | "pug-walk": "^2.0.0" 1963 | } 1964 | }, 1965 | "node_modules/pug-load": { 1966 | "version": "3.0.0", 1967 | "resolved": "https://registry.npmjs.org/pug-load/-/pug-load-3.0.0.tgz", 1968 | "integrity": "sha512-OCjTEnhLWZBvS4zni/WUMjH2YSUosnsmjGBB1An7CsKQarYSWQ0GCVyd4eQPMFJqZ8w9xgs01QdiZXKVjk92EQ==", 1969 | "dependencies": { 1970 | "object-assign": "^4.1.1", 1971 | "pug-walk": "^2.0.0" 1972 | } 1973 | }, 1974 | "node_modules/pug-parser": { 1975 | "version": "6.0.0", 1976 | "resolved": "https://registry.npmjs.org/pug-parser/-/pug-parser-6.0.0.tgz", 1977 | "integrity": "sha512-ukiYM/9cH6Cml+AOl5kETtM9NR3WulyVP2y4HOU45DyMim1IeP/OOiyEWRr6qk5I5klpsBnbuHpwKmTx6WURnw==", 1978 | "dependencies": { 1979 | "pug-error": "^2.0.0", 1980 | "token-stream": "1.0.0" 1981 | } 1982 | }, 1983 | "node_modules/pug-runtime": { 1984 | "version": "3.0.1", 1985 | "resolved": "https://registry.npmjs.org/pug-runtime/-/pug-runtime-3.0.1.tgz", 1986 | "integrity": "sha512-L50zbvrQ35TkpHwv0G6aLSuueDRwc/97XdY8kL3tOT0FmhgG7UypU3VztfV/LATAvmUfYi4wNxSajhSAeNN+Kg==" 1987 | }, 1988 | "node_modules/pug-strip-comments": { 1989 | "version": "2.0.0", 1990 | "resolved": "https://registry.npmjs.org/pug-strip-comments/-/pug-strip-comments-2.0.0.tgz", 1991 | "integrity": "sha512-zo8DsDpH7eTkPHCXFeAk1xZXJbyoTfdPlNR0bK7rpOMuhBYb0f5qUVCO1xlsitYd3w5FQTK7zpNVKb3rZoUrrQ==", 1992 | "dependencies": { 1993 | "pug-error": "^2.0.0" 1994 | } 1995 | }, 1996 | "node_modules/pug-walk": { 1997 | "version": "2.0.0", 1998 | "resolved": "https://registry.npmjs.org/pug-walk/-/pug-walk-2.0.0.tgz", 1999 | "integrity": "sha512-yYELe9Q5q9IQhuvqsZNwA5hfPkMJ8u92bQLIMcsMxf/VADjNtEYptU+inlufAFYcWdHlwNfZOEnOOQrZrcyJCQ==" 2000 | }, 2001 | "node_modules/punycode": { 2002 | "version": "2.3.1", 2003 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", 2004 | "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", 2005 | "engines": { 2006 | "node": ">=6" 2007 | } 2008 | }, 2009 | "node_modules/qs": { 2010 | "version": "6.13.0", 2011 | "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz", 2012 | "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==", 2013 | "dependencies": { 2014 | "side-channel": "^1.0.6" 2015 | }, 2016 | "engines": { 2017 | "node": ">=0.6" 2018 | }, 2019 | "funding": { 2020 | "url": "https://github.com/sponsors/ljharb" 2021 | } 2022 | }, 2023 | "node_modules/range-parser": { 2024 | "version": "1.2.1", 2025 | "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", 2026 | "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", 2027 | "engines": { 2028 | "node": ">= 0.6" 2029 | } 2030 | }, 2031 | "node_modules/raw-body": { 2032 | "version": "2.5.2", 2033 | "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz", 2034 | "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==", 2035 | "dependencies": { 2036 | "bytes": "3.1.2", 2037 | "http-errors": "2.0.0", 2038 | "iconv-lite": "0.4.24", 2039 | "unpipe": "1.0.0" 2040 | }, 2041 | "engines": { 2042 | "node": ">= 0.8" 2043 | } 2044 | }, 2045 | "node_modules/readable-stream": { 2046 | "version": "3.6.2", 2047 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", 2048 | "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", 2049 | "license": "MIT", 2050 | "dependencies": { 2051 | "inherits": "^2.0.3", 2052 | "string_decoder": "^1.1.1", 2053 | "util-deprecate": "^1.0.1" 2054 | }, 2055 | "engines": { 2056 | "node": ">= 6" 2057 | } 2058 | }, 2059 | "node_modules/readdirp": { 2060 | "version": "3.6.0", 2061 | "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", 2062 | "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", 2063 | "dev": true, 2064 | "dependencies": { 2065 | "picomatch": "^2.2.1" 2066 | }, 2067 | "engines": { 2068 | "node": ">=8.10.0" 2069 | } 2070 | }, 2071 | "node_modules/resolve": { 2072 | "version": "1.22.10", 2073 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz", 2074 | "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==", 2075 | "dependencies": { 2076 | "is-core-module": "^2.16.0", 2077 | "path-parse": "^1.0.7", 2078 | "supports-preserve-symlinks-flag": "^1.0.0" 2079 | }, 2080 | "bin": { 2081 | "resolve": "bin/resolve" 2082 | }, 2083 | "engines": { 2084 | "node": ">= 0.4" 2085 | }, 2086 | "funding": { 2087 | "url": "https://github.com/sponsors/ljharb" 2088 | } 2089 | }, 2090 | "node_modules/rimraf": { 2091 | "version": "3.0.2", 2092 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", 2093 | "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", 2094 | "deprecated": "Rimraf versions prior to v4 are no longer supported", 2095 | "license": "ISC", 2096 | "dependencies": { 2097 | "glob": "^7.1.3" 2098 | }, 2099 | "bin": { 2100 | "rimraf": "bin.js" 2101 | }, 2102 | "funding": { 2103 | "url": "https://github.com/sponsors/isaacs" 2104 | } 2105 | }, 2106 | "node_modules/safe-buffer": { 2107 | "version": "5.2.1", 2108 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 2109 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 2110 | "funding": [ 2111 | { 2112 | "type": "github", 2113 | "url": "https://github.com/sponsors/feross" 2114 | }, 2115 | { 2116 | "type": "patreon", 2117 | "url": "https://www.patreon.com/feross" 2118 | }, 2119 | { 2120 | "type": "consulting", 2121 | "url": "https://feross.org/support" 2122 | } 2123 | ] 2124 | }, 2125 | "node_modules/safer-buffer": { 2126 | "version": "2.1.2", 2127 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 2128 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" 2129 | }, 2130 | "node_modules/semver": { 2131 | "version": "7.7.1", 2132 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", 2133 | "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", 2134 | "bin": { 2135 | "semver": "bin/semver.js" 2136 | }, 2137 | "engines": { 2138 | "node": ">=10" 2139 | } 2140 | }, 2141 | "node_modules/send": { 2142 | "version": "0.19.0", 2143 | "resolved": "https://registry.npmjs.org/send/-/send-0.19.0.tgz", 2144 | "integrity": "sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==", 2145 | "dependencies": { 2146 | "debug": "2.6.9", 2147 | "depd": "2.0.0", 2148 | "destroy": "1.2.0", 2149 | "encodeurl": "~1.0.2", 2150 | "escape-html": "~1.0.3", 2151 | "etag": "~1.8.1", 2152 | "fresh": "0.5.2", 2153 | "http-errors": "2.0.0", 2154 | "mime": "1.6.0", 2155 | "ms": "2.1.3", 2156 | "on-finished": "2.4.1", 2157 | "range-parser": "~1.2.1", 2158 | "statuses": "2.0.1" 2159 | }, 2160 | "engines": { 2161 | "node": ">= 0.8.0" 2162 | } 2163 | }, 2164 | "node_modules/send/node_modules/encodeurl": { 2165 | "version": "1.0.2", 2166 | "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", 2167 | "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", 2168 | "engines": { 2169 | "node": ">= 0.8" 2170 | } 2171 | }, 2172 | "node_modules/send/node_modules/ms": { 2173 | "version": "2.1.3", 2174 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 2175 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 2176 | }, 2177 | "node_modules/serve-static": { 2178 | "version": "1.16.2", 2179 | "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.2.tgz", 2180 | "integrity": "sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw==", 2181 | "dependencies": { 2182 | "encodeurl": "~2.0.0", 2183 | "escape-html": "~1.0.3", 2184 | "parseurl": "~1.3.3", 2185 | "send": "0.19.0" 2186 | }, 2187 | "engines": { 2188 | "node": ">= 0.8.0" 2189 | } 2190 | }, 2191 | "node_modules/set-blocking": { 2192 | "version": "2.0.0", 2193 | "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", 2194 | "integrity": "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==", 2195 | "license": "ISC" 2196 | }, 2197 | "node_modules/setprototypeof": { 2198 | "version": "1.2.0", 2199 | "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", 2200 | "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" 2201 | }, 2202 | "node_modules/side-channel": { 2203 | "version": "1.1.0", 2204 | "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", 2205 | "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", 2206 | "dependencies": { 2207 | "es-errors": "^1.3.0", 2208 | "object-inspect": "^1.13.3", 2209 | "side-channel-list": "^1.0.0", 2210 | "side-channel-map": "^1.0.1", 2211 | "side-channel-weakmap": "^1.0.2" 2212 | }, 2213 | "engines": { 2214 | "node": ">= 0.4" 2215 | }, 2216 | "funding": { 2217 | "url": "https://github.com/sponsors/ljharb" 2218 | } 2219 | }, 2220 | "node_modules/side-channel-list": { 2221 | "version": "1.0.0", 2222 | "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", 2223 | "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", 2224 | "dependencies": { 2225 | "es-errors": "^1.3.0", 2226 | "object-inspect": "^1.13.3" 2227 | }, 2228 | "engines": { 2229 | "node": ">= 0.4" 2230 | }, 2231 | "funding": { 2232 | "url": "https://github.com/sponsors/ljharb" 2233 | } 2234 | }, 2235 | "node_modules/side-channel-map": { 2236 | "version": "1.0.1", 2237 | "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", 2238 | "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", 2239 | "dependencies": { 2240 | "call-bound": "^1.0.2", 2241 | "es-errors": "^1.3.0", 2242 | "get-intrinsic": "^1.2.5", 2243 | "object-inspect": "^1.13.3" 2244 | }, 2245 | "engines": { 2246 | "node": ">= 0.4" 2247 | }, 2248 | "funding": { 2249 | "url": "https://github.com/sponsors/ljharb" 2250 | } 2251 | }, 2252 | "node_modules/side-channel-weakmap": { 2253 | "version": "1.0.2", 2254 | "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", 2255 | "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", 2256 | "dependencies": { 2257 | "call-bound": "^1.0.2", 2258 | "es-errors": "^1.3.0", 2259 | "get-intrinsic": "^1.2.5", 2260 | "object-inspect": "^1.13.3", 2261 | "side-channel-map": "^1.0.1" 2262 | }, 2263 | "engines": { 2264 | "node": ">= 0.4" 2265 | }, 2266 | "funding": { 2267 | "url": "https://github.com/sponsors/ljharb" 2268 | } 2269 | }, 2270 | "node_modules/sift": { 2271 | "version": "17.1.3", 2272 | "resolved": "https://registry.npmjs.org/sift/-/sift-17.1.3.tgz", 2273 | "integrity": "sha512-Rtlj66/b0ICeFzYTuNvX/EF1igRbbnGSvEyT79McoZa/DeGhMyC5pWKOEsZKnpkqtSeovd5FL/bjHWC3CIIvCQ==" 2274 | }, 2275 | "node_modules/signal-exit": { 2276 | "version": "3.0.7", 2277 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", 2278 | "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", 2279 | "license": "ISC" 2280 | }, 2281 | "node_modules/simple-update-notifier": { 2282 | "version": "2.0.0", 2283 | "resolved": "https://registry.npmjs.org/simple-update-notifier/-/simple-update-notifier-2.0.0.tgz", 2284 | "integrity": "sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w==", 2285 | "dev": true, 2286 | "dependencies": { 2287 | "semver": "^7.5.3" 2288 | }, 2289 | "engines": { 2290 | "node": ">=10" 2291 | } 2292 | }, 2293 | "node_modules/sparse-bitfield": { 2294 | "version": "3.0.3", 2295 | "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", 2296 | "integrity": "sha512-kvzhi7vqKTfkh0PZU+2D2PIllw2ymqJKujUcyPMd9Y75Nv4nPbGJZXNhxsgdQab2BmlDct1YnfQCguEvHr7VsQ==", 2297 | "dependencies": { 2298 | "memory-pager": "^1.0.2" 2299 | } 2300 | }, 2301 | "node_modules/statuses": { 2302 | "version": "2.0.1", 2303 | "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", 2304 | "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", 2305 | "engines": { 2306 | "node": ">= 0.8" 2307 | } 2308 | }, 2309 | "node_modules/string_decoder": { 2310 | "version": "1.3.0", 2311 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", 2312 | "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", 2313 | "license": "MIT", 2314 | "dependencies": { 2315 | "safe-buffer": "~5.2.0" 2316 | } 2317 | }, 2318 | "node_modules/string-width": { 2319 | "version": "4.2.3", 2320 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 2321 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 2322 | "license": "MIT", 2323 | "dependencies": { 2324 | "emoji-regex": "^8.0.0", 2325 | "is-fullwidth-code-point": "^3.0.0", 2326 | "strip-ansi": "^6.0.1" 2327 | }, 2328 | "engines": { 2329 | "node": ">=8" 2330 | } 2331 | }, 2332 | "node_modules/strip-ansi": { 2333 | "version": "6.0.1", 2334 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 2335 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 2336 | "license": "MIT", 2337 | "dependencies": { 2338 | "ansi-regex": "^5.0.1" 2339 | }, 2340 | "engines": { 2341 | "node": ">=8" 2342 | } 2343 | }, 2344 | "node_modules/supports-color": { 2345 | "version": "5.5.0", 2346 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 2347 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 2348 | "dev": true, 2349 | "dependencies": { 2350 | "has-flag": "^3.0.0" 2351 | }, 2352 | "engines": { 2353 | "node": ">=4" 2354 | } 2355 | }, 2356 | "node_modules/supports-preserve-symlinks-flag": { 2357 | "version": "1.0.0", 2358 | "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", 2359 | "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", 2360 | "engines": { 2361 | "node": ">= 0.4" 2362 | }, 2363 | "funding": { 2364 | "url": "https://github.com/sponsors/ljharb" 2365 | } 2366 | }, 2367 | "node_modules/tar": { 2368 | "version": "6.2.1", 2369 | "resolved": "https://registry.npmjs.org/tar/-/tar-6.2.1.tgz", 2370 | "integrity": "sha512-DZ4yORTwrbTj/7MZYq2w+/ZFdI6OZ/f9SFHR+71gIVUZhOQPHzVCLpvRnPgyaMpfWxxk/4ONva3GQSyNIKRv6A==", 2371 | "license": "ISC", 2372 | "dependencies": { 2373 | "chownr": "^2.0.0", 2374 | "fs-minipass": "^2.0.0", 2375 | "minipass": "^5.0.0", 2376 | "minizlib": "^2.1.1", 2377 | "mkdirp": "^1.0.3", 2378 | "yallist": "^4.0.0" 2379 | }, 2380 | "engines": { 2381 | "node": ">=10" 2382 | } 2383 | }, 2384 | "node_modules/to-regex-range": { 2385 | "version": "5.0.1", 2386 | "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", 2387 | "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", 2388 | "dev": true, 2389 | "dependencies": { 2390 | "is-number": "^7.0.0" 2391 | }, 2392 | "engines": { 2393 | "node": ">=8.0" 2394 | } 2395 | }, 2396 | "node_modules/toidentifier": { 2397 | "version": "1.0.1", 2398 | "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", 2399 | "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", 2400 | "engines": { 2401 | "node": ">=0.6" 2402 | } 2403 | }, 2404 | "node_modules/token-stream": { 2405 | "version": "1.0.0", 2406 | "resolved": "https://registry.npmjs.org/token-stream/-/token-stream-1.0.0.tgz", 2407 | "integrity": "sha512-VSsyNPPW74RpHwR8Fc21uubwHY7wMDeJLys2IX5zJNih+OnAnaifKHo+1LHT7DAdloQ7apeaaWg8l7qnf/TnEg==" 2408 | }, 2409 | "node_modules/touch": { 2410 | "version": "3.1.1", 2411 | "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.1.tgz", 2412 | "integrity": "sha512-r0eojU4bI8MnHr8c5bNo7lJDdI2qXlWWJk6a9EAFG7vbhTjElYhBVS3/miuE0uOuoLdb8Mc/rVfsmm6eo5o9GA==", 2413 | "dev": true, 2414 | "bin": { 2415 | "nodetouch": "bin/nodetouch.js" 2416 | } 2417 | }, 2418 | "node_modules/tr46": { 2419 | "version": "5.1.0", 2420 | "resolved": "https://registry.npmjs.org/tr46/-/tr46-5.1.0.tgz", 2421 | "integrity": "sha512-IUWnUK7ADYR5Sl1fZlO1INDUhVhatWl7BtJWsIhwJ0UAK7ilzzIa8uIqOO/aYVWHZPJkKbEL+362wrzoeRF7bw==", 2422 | "dependencies": { 2423 | "punycode": "^2.3.1" 2424 | }, 2425 | "engines": { 2426 | "node": ">=18" 2427 | } 2428 | }, 2429 | "node_modules/type-is": { 2430 | "version": "1.6.18", 2431 | "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", 2432 | "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", 2433 | "dependencies": { 2434 | "media-typer": "0.3.0", 2435 | "mime-types": "~2.1.24" 2436 | }, 2437 | "engines": { 2438 | "node": ">= 0.6" 2439 | } 2440 | }, 2441 | "node_modules/undefsafe": { 2442 | "version": "2.0.5", 2443 | "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.5.tgz", 2444 | "integrity": "sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==", 2445 | "dev": true 2446 | }, 2447 | "node_modules/unpipe": { 2448 | "version": "1.0.0", 2449 | "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", 2450 | "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", 2451 | "engines": { 2452 | "node": ">= 0.8" 2453 | } 2454 | }, 2455 | "node_modules/util-deprecate": { 2456 | "version": "1.0.2", 2457 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 2458 | "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", 2459 | "license": "MIT" 2460 | }, 2461 | "node_modules/utils-merge": { 2462 | "version": "1.0.1", 2463 | "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", 2464 | "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", 2465 | "engines": { 2466 | "node": ">= 0.4.0" 2467 | } 2468 | }, 2469 | "node_modules/vary": { 2470 | "version": "1.1.2", 2471 | "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", 2472 | "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", 2473 | "engines": { 2474 | "node": ">= 0.8" 2475 | } 2476 | }, 2477 | "node_modules/void-elements": { 2478 | "version": "3.1.0", 2479 | "resolved": "https://registry.npmjs.org/void-elements/-/void-elements-3.1.0.tgz", 2480 | "integrity": "sha512-Dhxzh5HZuiHQhbvTW9AMetFfBHDMYpo23Uo9btPXgdYP+3T5S+p+jgNy7spra+veYhBP2dCSgxR/i2Y02h5/6w==", 2481 | "engines": { 2482 | "node": ">=0.10.0" 2483 | } 2484 | }, 2485 | "node_modules/webidl-conversions": { 2486 | "version": "7.0.0", 2487 | "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz", 2488 | "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==", 2489 | "engines": { 2490 | "node": ">=12" 2491 | } 2492 | }, 2493 | "node_modules/whatwg-url": { 2494 | "version": "14.2.0", 2495 | "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-14.2.0.tgz", 2496 | "integrity": "sha512-De72GdQZzNTUBBChsXueQUnPKDkg/5A5zp7pFDuQAj5UFoENpiACU0wlCvzpAGnTkj++ihpKwKyYewn/XNUbKw==", 2497 | "dependencies": { 2498 | "tr46": "^5.1.0", 2499 | "webidl-conversions": "^7.0.0" 2500 | }, 2501 | "engines": { 2502 | "node": ">=18" 2503 | } 2504 | }, 2505 | "node_modules/wide-align": { 2506 | "version": "1.1.5", 2507 | "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.5.tgz", 2508 | "integrity": "sha512-eDMORYaPNZ4sQIuuYPDHdQvf4gyCF9rEEV/yPxGfwPkRodwEgiMUUXTx/dex+Me0wxx53S+NgUHaP7y3MGlDmg==", 2509 | "license": "ISC", 2510 | "dependencies": { 2511 | "string-width": "^1.0.2 || 2 || 3 || 4" 2512 | } 2513 | }, 2514 | "node_modules/with": { 2515 | "version": "7.0.2", 2516 | "resolved": "https://registry.npmjs.org/with/-/with-7.0.2.tgz", 2517 | "integrity": "sha512-RNGKj82nUPg3g5ygxkQl0R937xLyho1J24ItRCBTr/m1YnZkzJy1hUiHUJrc/VlsDQzsCnInEGSg3bci0Lmd4w==", 2518 | "dependencies": { 2519 | "@babel/parser": "^7.9.6", 2520 | "@babel/types": "^7.9.6", 2521 | "assert-never": "^1.2.1", 2522 | "babel-walk": "3.0.0-canary-5" 2523 | }, 2524 | "engines": { 2525 | "node": ">= 10.0.0" 2526 | } 2527 | }, 2528 | "node_modules/wrappy": { 2529 | "version": "1.0.2", 2530 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 2531 | "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", 2532 | "license": "ISC" 2533 | }, 2534 | "node_modules/yallist": { 2535 | "version": "4.0.0", 2536 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", 2537 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", 2538 | "license": "ISC" 2539 | } 2540 | } 2541 | } 2542 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "backend-template", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "type": "module", 7 | "scripts": { 8 | "dev": "nodemon index.js", 9 | "start": "node index.js" 10 | }, 11 | "keywords": [], 12 | "author": "Kaml Mehamednur", 13 | "license": "ISC", 14 | "dependencies": { 15 | "axios": "^1.8.4", 16 | "bcrypt": "^5.1.1", 17 | "cors": "^2.8.5", 18 | "dotenv": "^16.4.7", 19 | "express": "^4.21.2", 20 | "helmet": "^8.0.0", 21 | "jsonwebtoken": "^9.0.2", 22 | "mongoose": "^8.12.1", 23 | "pug": "^3.0.3" 24 | }, 25 | "devDependencies": { 26 | "morgan": "^1.10.0", 27 | "nodemon": "^3.1.9" 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /public/css/style.css: -------------------------------------------------------------------------------- 1 | body { 2 | font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, 3 | "Helvetica Neue", Arial, sans-serif; 4 | line-height: 1.6; 5 | max-width: 1000px; 6 | margin: 0 auto; 7 | padding: 20px; 8 | } 9 | 10 | .api-title { 11 | color: #2c3e50; 12 | border-bottom: 2px solid #3498db; 13 | padding-bottom: 10px; 14 | } 15 | 16 | .endpoint { 17 | background: #f8f9fa; 18 | padding: 15px; 19 | margin: 10px 0; 20 | border-radius: 5px; 21 | box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); 22 | } 23 | 24 | .method { 25 | display: inline-block; 26 | padding: 4px 8px; 27 | border-radius: 4px; 28 | color: white; 29 | font-weight: bold; 30 | margin-right: 10px; 31 | } 32 | 33 | .get { 34 | background: #28a745; 35 | } 36 | .post { 37 | background: #007bff; 38 | } 39 | .put { 40 | background: #ffc107; 41 | color: #000; 42 | } 43 | .delete { 44 | background: #dc3545; 45 | } 46 | 47 | code { 48 | background: #e9ecef; 49 | padding: 2px 5px; 50 | border-radius: 3px; 51 | font-family: "Courier New", Courier, monospace; 52 | } 53 | 54 | pre { 55 | background: #f8f9fa; 56 | padding: 15px; 57 | border-radius: 5px; 58 | overflow-x: auto; 59 | } 60 | 61 | pre:hover { 62 | cursor: pointer; 63 | } -------------------------------------------------------------------------------- /public/js/main.js: -------------------------------------------------------------------------------- 1 | document.addEventListener("DOMContentLoaded", () => { 2 | // Add copy functionality to code blocks 3 | document.querySelectorAll("pre code").forEach((block) => { 4 | block.addEventListener("click", () => { 5 | const text = block.textContent; 6 | navigator.clipboard.writeText(text).then(() => { 7 | // Show a brief "Copied!" message 8 | const message = document.createElement("div"); 9 | message.textContent = "Copied!"; 10 | message.style.cssText = ` 11 | position: fixed; 12 | top: 20px; 13 | right: 20px; 14 | background: #28a745; 15 | color: white; 16 | padding: 10px 20px; 17 | border-radius: 5px; 18 | transition: opacity 0.3s; 19 | `; 20 | document.body.appendChild(message); 21 | setTimeout(() => { 22 | message.style.opacity = "0"; 23 | setTimeout(() => message.remove(), 300); 24 | }, 2000); 25 | }); 26 | }); 27 | }); 28 | }); -------------------------------------------------------------------------------- /routes/admin.routes.js: -------------------------------------------------------------------------------- 1 | import { Router } from "express"; 2 | import bcrypt from "bcrypt"; 3 | import jwt from "jsonwebtoken"; 4 | import Admin from "../models/Admin.js"; 5 | 6 | const adminRouter = new Router(); 7 | 8 | // Admin Registration Route 9 | adminRouter.post("/register", async (req, res) => { 10 | try { 11 | const { username, password } = req.body; 12 | 13 | if (!username || !password) { 14 | return res 15 | .status(400) 16 | .json({ message: "Username and password are required" }); 17 | } 18 | 19 | const existingAdmin = await Admin.findOne({ username }); 20 | if (existingAdmin) { 21 | return res 22 | .status(400) 23 | .json({ message: "Admin with this username already exists" }); 24 | } 25 | 26 | const hashedPassword = bcrypt.hash(password, 10); 27 | 28 | const newAdmin = new Admin({ 29 | username, 30 | password: hashedPassword, 31 | }); 32 | 33 | await newAdmin.save(); 34 | 35 | res.status(201).json({ message: "Admin created successfully" }); 36 | } catch (error) { 37 | console.error(error); 38 | res.status(500).json({ message: error.message }); 39 | } 40 | }); 41 | 42 | // Admin Login Route 43 | adminRouter.post("/login", async (req, res) => { 44 | try { 45 | const { username, password } = req.body; 46 | 47 | if (!username || !password) { 48 | return res 49 | .status(400) 50 | .json({ message: "Username and password are required" }); 51 | } 52 | 53 | const admin = await Admin.findOne({ username }); 54 | if (!admin) { 55 | return res.status(401).json({ message: "Invalid credentials" }); 56 | } 57 | 58 | const isMatch = bcrypt.compare(password, admin.password); 59 | if (!isMatch) { 60 | return res.status(401).json({ message: "Invalid credentials" }); 61 | } 62 | 63 | const token = jwt.sign({ id: admin._id }, process.env.JWT_SECRET, { 64 | expiresIn: "1d", 65 | }); 66 | 67 | res.json({ token }); 68 | } catch (error) { 69 | res.status(500).json({ message: error.message }); 70 | } 71 | }); 72 | 73 | export default adminRouter; 74 | -------------------------------------------------------------------------------- /routes/bookReview.routes.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | const router = express.Router(); 3 | import BookReview from "../models/BookReview.js"; 4 | import { authenticateAdmin } from "../middleware/auth.js"; 5 | 6 | router.get("/", async (req, res) => { 7 | try { 8 | const bookReviews = await BookReview.find().sort({ createdAt: -1 }); 9 | res.json(bookReviews); 10 | } catch (err) { 11 | res.status(500).json({ message: err.message }); 12 | } 13 | }); 14 | 15 | router.get("/:bookId", async (req, res) => { 16 | try { 17 | const bookReview = await BookReview.findById(req.params.bookId); 18 | if (!bookReview) { 19 | return res.status(404).json({ message: "Book review not found" }); 20 | } 21 | res.json(bookReview); 22 | } catch (err) { 23 | res.status(500).json({ message: err.message }); 24 | } 25 | }); 26 | 27 | router.post("/", authenticateAdmin, async (req, res) => { 28 | try { 29 | const bookData = req.body.bookData; 30 | 31 | if (!bookData) { 32 | return res.status(400).json({ message: "Failed to fetch book data" }); 33 | } 34 | 35 | const bookReview = new BookReview({ 36 | bookId: req.body.bookId, 37 | title: bookData.title, 38 | subTitle: bookData.subTitle || "N/A", 39 | authors: bookData.authors || "N/A", 40 | thumbnail: bookData?.imageLinks?.thumbnail || bookData?.thumbnail, 41 | genres: Array.isArray(bookData.categories) 42 | ? bookData.categories.join(", ") 43 | : "N/A", 44 | descriptionShort: bookData.description || req.body.descriptionShort, 45 | howChangedMe: req.body.howChangedMe, 46 | whoShouldRead: req.body.whoShouldRead, 47 | impressions: req.body.impressions, 48 | rating: req.body.rating, 49 | }); 50 | 51 | const newBookReview = await bookReview.save(); 52 | res.status(201).json(newBookReview); 53 | } catch (err) { 54 | res.status(400).json({ message: err.message }); 55 | } 56 | }); 57 | 58 | router.put("/:id", authenticateAdmin, async (req, res) => { 59 | try { 60 | const review = await BookReview.findById(req.params.id); 61 | if (!review) return res.status(404).json({ message: "Review not found" }); 62 | 63 | const fields = [ 64 | "descriptionShort", 65 | "howChangedMe", 66 | "whoShouldRead", 67 | "impressions", 68 | "rating", 69 | ]; 70 | fields.forEach((field) => { 71 | if (req.body[field] !== undefined) review[field] = req.body[field]; 72 | }); 73 | 74 | const updated = await review.save(); 75 | res.json(updated); 76 | } catch (err) { 77 | res.status(500).json({ message: err.message }); 78 | } 79 | }); 80 | 81 | router.delete("/:id", authenticateAdmin, async (req, res) => { 82 | try { 83 | const deletedBookReview = await BookReview.findByIdAndDelete(req.params.id); 84 | if (!deletedBookReview) { 85 | return res.status(404).json({ message: "Book review not found" }); 86 | } 87 | res.json({ message: "Book review deleted" }); 88 | } catch (err) { 89 | res.status(500).json({ message: err.message }); 90 | } 91 | }); 92 | 93 | export default router; 94 | -------------------------------------------------------------------------------- /routes/message.routes.js: -------------------------------------------------------------------------------- 1 | import express from "express"; 2 | const router = express.Router(); 3 | import Message from "../models/Message.js"; 4 | import { authenticateAdmin } from "../middleware/auth.js"; 5 | 6 | router.get("/", authenticateAdmin, async (req, res) => { 7 | try { 8 | const messages = await Message.find().sort({ createdAt: -1 }); 9 | res.json(messages); 10 | } catch (err) { 11 | res.status(500).json({ message: err.message }); 12 | } 13 | }); 14 | 15 | router.get("/testimonials", async (req, res) => { 16 | try { 17 | const testimonials = await Message.find({ type: "testimonial" }); 18 | res.json(testimonials); 19 | } catch (err) { 20 | res.status(500).json({ message: err.message }); 21 | } 22 | }); 23 | 24 | router.post("/", async (req, res) => { 25 | const message = new Message(req.body); 26 | try { 27 | const newMessage = await message.save(); 28 | res.status(201).json(newMessage); 29 | } catch (err) { 30 | res.status(400).json({ message: err.message }); 31 | } 32 | }); 33 | 34 | router.patch("/:id", authenticateAdmin, async (req, res) => { 35 | try { 36 | const updatedMessage = await Message.findByIdAndUpdate( 37 | req.params.id, 38 | req.body, 39 | { new: true } 40 | ); 41 | if (!updatedMessage) { 42 | return res.status(404).json({ message: "Message not found" }); 43 | } 44 | res.json(updatedMessage); 45 | } catch (err) { 46 | res.status(400).json({ message: err.message }); 47 | } 48 | }); 49 | 50 | router.delete("/:id", authenticateAdmin, async (req, res) => { 51 | try { 52 | const deletedMessage = await Message.findByIdAndDelete(req.params.id); 53 | if (!deletedMessage) { 54 | return res.status(404).json({ message: "Message not found" }); 55 | } 56 | res.json({ message: "Message deleted" }); 57 | } catch (err) { 58 | res.status(500).json({ message: err.message }); 59 | } 60 | }); 61 | 62 | export default router; 63 | -------------------------------------------------------------------------------- /routes/project.routes.js: -------------------------------------------------------------------------------- 1 | import { Router } from "express"; 2 | import Project from "../models/Project.js"; 3 | import { authenticateAdmin } from "../middleware/auth.js"; 4 | const router = Router(); 5 | 6 | // GET all projects (public) 7 | router.get("/", async (req, res) => { 8 | try { 9 | const projects = await Project.find().sort({ order: 1 }); 10 | res.json(projects); 11 | } catch (err) { 12 | res.status(500).json({ message: err.message }); 13 | } 14 | }); 15 | 16 | // GET a single project (public) 17 | router.get("/:id", async (req, res) => { 18 | try { 19 | const project = await Project.findById(req.params.id); 20 | if (!project) { 21 | return res.status(404).json({ message: "Project not found" }); 22 | } 23 | res.json(project); 24 | } catch (err) { 25 | res.status(500).json({ message: err.message }); 26 | } 27 | }); 28 | 29 | // POST a new project (admin only) 30 | router.post("/", authenticateAdmin, async (req, res) => { 31 | const project = new Project(req.body); 32 | try { 33 | const newProject = await project.save(); 34 | res.status(201).json(newProject); 35 | } catch (err) { 36 | res.status(400).json({ message: err.message }); 37 | } 38 | }); 39 | 40 | // PATCH update a project (admin only) 41 | router.patch("/:id", authenticateAdmin, async (req, res) => { 42 | try { 43 | const updatedProject = await Project.findByIdAndUpdate( 44 | req.params.id, 45 | req.body, 46 | { new: true } 47 | ); 48 | if (!updatedProject) { 49 | return res.status(404).json({ message: "Project not found" }); 50 | } 51 | res.json(updatedProject); 52 | } catch (err) { 53 | res.status(400).json({ message: err.message }); 54 | } 55 | }); 56 | 57 | // DELETE a project (admin only) 58 | router.delete("/:id", authenticateAdmin, async (req, res) => { 59 | try { 60 | const deletedProject = await Project.findByIdAndDelete(req.params.id); 61 | if (!deletedProject) { 62 | return res.status(404).json({ message: "Project not found" }); 63 | } 64 | res.json({ message: "Project deleted" }); 65 | } catch (err) { 66 | res.status(500).json({ message: err.message }); 67 | } 68 | }); 69 | 70 | export default router; 71 | -------------------------------------------------------------------------------- /views/index.pug: -------------------------------------------------------------------------------- 1 | doctype html 2 | html(lang="en") 3 | head 4 | title API Documentation 5 | link(rel="stylesheet" href="/css/style.css") 6 | 7 | body 8 | h1.api-title API Documentation 9 | 10 | h2 Base URL 11 | p All endpoints are prefixed with: 12 | code http://localhost:4000/api 13 | 14 | h2 Authentication 15 | p Currently, no authentication is required for the API endpoints. 16 | 17 | h2 Endpoints 18 | 19 | h3 Health Check 20 | .endpoint 21 | span.method.get GET 22 | code /health 23 | p Returns the API health status. 24 | h4 Response 25 | pre 26 | code 27 | | { 28 | | "status": "OK" 29 | | } 30 | 31 | h3 Users 32 | 33 | .endpoint 34 | span.method.get GET 35 | code /users 36 | p Retrieve all users 37 | h4 Response 38 | pre 39 | code 40 | | [ 41 | | { 42 | | "_id": "string", 43 | | "name": "string", 44 | | "email": "string", 45 | | "createdAt": "date" 46 | | } 47 | | ] 48 | 49 | .endpoint 50 | span.method.get GET 51 | code /users/:id 52 | p Retrieve a specific user by ID 53 | h4 Response 54 | pre 55 | code 56 | | { 57 | | "_id": "string", 58 | | "name": "string", 59 | | "email": "string", 60 | | "createdAt": "date" 61 | | } 62 | 63 | .endpoint 64 | span.method.post POST 65 | code /users 66 | p Create a new user 67 | h4 Request Body 68 | table 69 | tr 70 | th Field 71 | th Type 72 | th Required 73 | th Description 74 | tr 75 | td name 76 | td string 77 | td Yes 78 | td User's full name 79 | tr 80 | td email 81 | td string 82 | td Yes 83 | td User's email address 84 | h4 Response 85 | pre 86 | code 87 | | { 88 | | "_id": "string", 89 | | "name": "string", 90 | | "email": "string", 91 | | "createdAt": "date" 92 | | } 93 | 94 | .endpoint 95 | span.method.put PUT 96 | code /users/:id 97 | p Update an existing user 98 | h4 Request Body 99 | table 100 | tr 101 | th Field 102 | th Type 103 | th Required 104 | th Description 105 | tr 106 | td name 107 | td string 108 | td No 109 | td User's full name 110 | tr 111 | td email 112 | td string 113 | td No 114 | td User's email address 115 | h4 Response 116 | pre 117 | code 118 | | { 119 | | "_id": "string", 120 | | "name": "string", 121 | | "email": "string", 122 | | "createdAt": "date" 123 | | } 124 | 125 | .endpoint 126 | span.method.delete DELETE 127 | code /users/:id 128 | p Delete a user 129 | h4 Response 130 | p Returns status code 204 with no content on success. 131 | 132 | h2 Error Responses 133 | p The API returns standard HTTP status codes and JSON error messages: 134 | pre 135 | code 136 | | { 137 | | "error": "Error Type", 138 | | "message": "Error description" 139 | | } 140 | 141 | script(src="/js/main.js") --------------------------------------------------------------------------------