ShieldX commited on
Commit
c0fb352
·
verified ·
1 Parent(s): 663286a

Upload 13 files

Browse files
.dockerignore ADDED
@@ -0,0 +1,4 @@
 
 
 
 
 
1
+ node_modules
2
+ .env
3
+ .git
4
+ npm-debug.log
Dockerfile ADDED
@@ -0,0 +1,21 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # Use the official lightweight Node.js image
2
+ FROM node:22-alpine
3
+
4
+ # Set the working directory inside the container
5
+ WORKDIR /app
6
+
7
+ # Copy package files and install dependencies
8
+ COPY package*.json ./
9
+ RUN npm install
10
+
11
+ # Copy the rest of the application code
12
+ COPY . .
13
+
14
+ # Hugging Face Spaces expose port 7860 by default
15
+ EXPOSE 7860
16
+
17
+ # Set the PORT environment variable for our server.js to pick up
18
+ ENV PORT=7860
19
+
20
+ # Command to start the app
21
+ CMD ["npm", "start"]
config/cloudinary.js ADDED
@@ -0,0 +1,27 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ // config/cloudinary.js
2
+ import { v2 as cloudinary } from 'cloudinary';
3
+ import { CloudinaryStorage } from 'multer-storage-cloudinary';
4
+ import multer from 'multer';
5
+ import dotenv from 'dotenv';
6
+
7
+ dotenv.config();
8
+
9
+ // Configure Cloudinary with your credentials
10
+ cloudinary.config({
11
+ cloud_name: process.env.CLOUDINARY_CLOUD_NAME,
12
+ api_key: process.env.CLOUDINARY_API_KEY,
13
+ api_secret: process.env.CLOUDINARY_API_SECRET,
14
+ });
15
+
16
+ // Setup the storage engine
17
+ const storage = new CloudinaryStorage({
18
+ cloudinary: cloudinary,
19
+ params: {
20
+ folder: 'house-of-ruqa-products', // The folder name inside your Cloudinary account
21
+ allowedFormats: ['jpeg', 'png', 'jpg', 'webp'],
22
+ transformation: [{ width: 1000, crop: 'limit' }], // Prevents massive file sizes
23
+ },
24
+ });
25
+
26
+ // Export the multer middleware so we can use it in our routes
27
+ export const upload = multer({ storage });
config/db.js ADDED
@@ -0,0 +1,12 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ // config/db.js
2
+ import mongoose from 'mongoose';
3
+
4
+ export const connectDB = async () => {
5
+ try {
6
+ const conn = await mongoose.connect(process.env.MONGODB_URI);
7
+ console.log(`🤎 MongoDB Connected: ${conn.connection.host}`);
8
+ } catch (error) {
9
+ console.error(`❌ Error connecting to MongoDB: ${error.message}`);
10
+ process.exit(1); // Exit process with failure
11
+ }
12
+ };
controllers/productController.js ADDED
@@ -0,0 +1,134 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ // controllers/productController.js
2
+ import Product from '../models/Product.js';
3
+
4
+ // @desc Create a new product & upload images
5
+ // @route POST /api/products
6
+ export const createProduct = async (req, res) => {
7
+ try {
8
+ // 1. Extract text data from the request body
9
+ const { name, price, description, fabric, category, subcategory } = req.body;
10
+
11
+ // sizes might come as a stringified array or single string depending on FormData,
12
+ // so we parse it safely.
13
+ let sizesArray = [];
14
+ if (req.body.sizes) {
15
+ sizesArray = Array.isArray(req.body.sizes) ? req.body.sizes : req.body.sizes.split(',');
16
+ }
17
+
18
+ // 2. Extract uploaded image URLs from Cloudinary
19
+ // req.files is populated by our Multer/Cloudinary middleware
20
+ if (!req.files || req.files.length === 0) {
21
+ return res.status(400).json({ message: 'At least one image is required' });
22
+ }
23
+
24
+ const imageUrls = req.files.map((file) => file.path);
25
+
26
+ // 3. Create the product in MongoDB
27
+ const product = new Product({
28
+ name,
29
+ price: Number(price),
30
+ description,
31
+ fabric,
32
+ category,
33
+ subcategory,
34
+ sizes: sizesArray,
35
+ images: imageUrls,
36
+ });
37
+
38
+ const savedProduct = await product.save();
39
+
40
+ // 4. Send success response back to the frontend
41
+ res.status(201).json({
42
+ success: true,
43
+ message: 'Product added to House of Ruqa catalog successfully',
44
+ product: savedProduct
45
+ });
46
+
47
+ } catch (error) {
48
+ console.error('Error creating product:', error);
49
+ res.status(500).json({ success: false, message: 'Server Error', error: error.message });
50
+ }
51
+ };
52
+
53
+ // @desc Get all products (For the storefront grid)
54
+ // @route GET /api/products
55
+ export const getProducts = async (req, res) => {
56
+ try {
57
+ // Optional filtering (e.g., /api/products?category=Women)
58
+ const filter = req.query.category ? { category: req.query.category } : {};
59
+
60
+ // Fetch from DB, newest first
61
+ const products = await Product.find(filter).sort({ createdAt: -1 });
62
+
63
+ res.status(200).json({ success: true, count: products.length, products });
64
+ } catch (error) {
65
+ res.status(500).json({ success: false, message: 'Server Error' });
66
+ }
67
+ };
68
+
69
+ // @desc Get single product by ID
70
+ // @route GET /api/products/:id
71
+ export const getProductById = async (req, res) => {
72
+ try {
73
+ const product = await Product.findById(req.params.id);
74
+ if (product) {
75
+ res.status(200).json({ success: true, product });
76
+ } else {
77
+ res.status(404).json({ success: false, message: 'Product not found' });
78
+ }
79
+ } catch (error) {
80
+ res.status(500).json({ success: false, message: 'Server Error' });
81
+ }
82
+ };
83
+
84
+ // @desc Update a product
85
+ // @route PUT /api/products/:id
86
+ export const updateProduct = async (req, res) => {
87
+ try {
88
+ const { name, price, description, fabric, category, subcategory } = req.body;
89
+
90
+ let sizesArray = [];
91
+ if (req.body.sizes) {
92
+ sizesArray = Array.isArray(req.body.sizes) ? req.body.sizes : req.body.sizes.split(',').map(s => s.trim());
93
+ }
94
+
95
+ // Find product and update (Note: For MVP, we are updating text/details. Image updates require a more complex Cloudinary sync)
96
+ const updatedProduct = await Product.findByIdAndUpdate(
97
+ req.params.id,
98
+ {
99
+ name,
100
+ price: Number(price),
101
+ description,
102
+ fabric,
103
+ category,
104
+ subcategory,
105
+ sizes: sizesArray,
106
+ },
107
+ { new: true, runValidators: true }
108
+ );
109
+
110
+ if (!updatedProduct) {
111
+ return res.status(404).json({ success: false, message: 'Product not found' });
112
+ }
113
+
114
+ res.status(200).json({ success: true, product: updatedProduct, message: 'Product updated successfully' });
115
+ } catch (error) {
116
+ res.status(500).json({ success: false, message: 'Server Error updating product' });
117
+ }
118
+ };
119
+
120
+ // @desc Delete a product
121
+ // @route DELETE /api/products/:id
122
+ export const deleteProduct = async (req, res) => {
123
+ try {
124
+ const product = await Product.findByIdAndDelete(req.params.id);
125
+
126
+ if (!product) {
127
+ return res.status(404).json({ success: false, message: 'Product not found' });
128
+ }
129
+
130
+ res.status(200).json({ success: true, message: 'Product deleted successfully' });
131
+ } catch (error) {
132
+ res.status(500).json({ success: false, message: 'Server Error deleting product' });
133
+ }
134
+ };
controllers/subcategoryController.js ADDED
@@ -0,0 +1,83 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import Subcategory from '../models/Subcategory.js';
2
+
3
+ // @desc Create a new subcategory with an image
4
+ // @route POST /api/subcategories
5
+ export const createSubcategory = async (req, res) => {
6
+ try {
7
+ const { name, parentCategory } = req.body;
8
+
9
+ // Ensure an image was uploaded via Cloudinary/Multer
10
+ if (!req.file) {
11
+ return res.status(400).json({ success: false, message: 'Subcategory image is required' });
12
+ }
13
+
14
+ const subcategory = new Subcategory({
15
+ name,
16
+ parentCategory,
17
+ image: req.file.path // Cloudinary secure URL
18
+ });
19
+
20
+ const savedSubcategory = await subcategory.save();
21
+ res.status(201).json({ success: true, subcategory: savedSubcategory });
22
+
23
+ } catch (error) {
24
+ res.status(500).json({ success: false, message: 'Server Error', error: error.message });
25
+ }
26
+ };
27
+
28
+ // @desc Get all subcategories (Optionally filter by parentCategory)
29
+ // @route GET /api/subcategories
30
+ export const getSubcategories = async (req, res) => {
31
+ try {
32
+ const filter = req.query.parentCategory ? { parentCategory: req.query.parentCategory } : {};
33
+ const subcategories = await Subcategory.find(filter).sort({ createdAt: -1 });
34
+
35
+ res.status(200).json({ success: true, subcategories });
36
+ } catch (error) {
37
+ res.status(500).json({ success: false, message: 'Server Error' });
38
+ }
39
+ };
40
+
41
+ // @desc Update a subcategory
42
+ // @route PUT /api/subcategories/:id
43
+ export const updateSubcategory = async (req, res) => {
44
+ try {
45
+ const { name, parentCategory } = req.body;
46
+ let updateData = { name, parentCategory };
47
+
48
+ // If the user uploaded a new image, update the URL
49
+ if (req.file) {
50
+ updateData.image = req.file.path;
51
+ }
52
+
53
+ const updatedSubcategory = await Subcategory.findByIdAndUpdate(
54
+ req.params.id,
55
+ updateData,
56
+ { new: true, runValidators: true }
57
+ );
58
+
59
+ if (!updatedSubcategory) {
60
+ return res.status(404).json({ success: false, message: 'Subcategory not found' });
61
+ }
62
+
63
+ res.status(200).json({ success: true, subcategory: updatedSubcategory, message: 'Updated successfully' });
64
+ } catch (error) {
65
+ res.status(500).json({ success: false, message: 'Server Error updating subcategory' });
66
+ }
67
+ };
68
+
69
+ // @desc Delete a subcategory
70
+ // @route DELETE /api/subcategories/:id
71
+ export const deleteSubcategory = async (req, res) => {
72
+ try {
73
+ const subcategory = await Subcategory.findByIdAndDelete(req.params.id);
74
+
75
+ if (!subcategory) {
76
+ return res.status(404).json({ success: false, message: 'Subcategory not found' });
77
+ }
78
+
79
+ res.status(200).json({ success: true, message: 'Subcategory deleted successfully' });
80
+ } catch (error) {
81
+ res.status(500).json({ success: false, message: 'Server Error deleting subcategory' });
82
+ }
83
+ };
models/Product.js ADDED
@@ -0,0 +1,47 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ // models/Product.js
2
+ import mongoose from 'mongoose';
3
+
4
+ const productSchema = new mongoose.Schema({
5
+ name: {
6
+ type: String,
7
+ required: [true, 'Product name is required'],
8
+ trim: true
9
+ },
10
+ price: {
11
+ type: Number,
12
+ required: [true, 'Product price is required']
13
+ },
14
+ description: {
15
+ type: String,
16
+ required: [true, 'Product description is required']
17
+ },
18
+ fabric: {
19
+ type: String,
20
+ default: 'Premium Blend'
21
+ },
22
+ category: {
23
+ type: String,
24
+ required: true,
25
+ enum: ['Women', 'Men'] // Strictly enforces these two categories
26
+ },
27
+ subcategory: {
28
+ type: String,
29
+ required: true
30
+ },
31
+ sizes: [{
32
+ type: String
33
+ }],
34
+ images: [{
35
+ type: String, // This will store the secure Cloudinary URLs
36
+ required: true
37
+ }],
38
+ isBestseller: {
39
+ type: Boolean,
40
+ default: false
41
+ }
42
+ }, {
43
+ timestamps: true // Automatically adds createdAt and updatedAt dates
44
+ });
45
+
46
+ const Product = mongoose.model('Product', productSchema);
47
+ export default Product;
models/Subcategory.js ADDED
@@ -0,0 +1,23 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import mongoose from 'mongoose';
2
+
3
+ const subcategorySchema = new mongoose.Schema({
4
+ name: {
5
+ type: String,
6
+ required: [true, 'Subcategory name is required'],
7
+ trim: true
8
+ },
9
+ parentCategory: {
10
+ type: String,
11
+ required: true,
12
+ enum: ['Women', 'Men'] // Links it strictly to your main edits
13
+ },
14
+ image: {
15
+ type: String, // Cloudinary URL for the visual bubble
16
+ required: true
17
+ }
18
+ }, {
19
+ timestamps: true
20
+ });
21
+
22
+ const Subcategory = mongoose.model('Subcategory', subcategorySchema);
23
+ export default Subcategory;
package-lock.json ADDED
@@ -0,0 +1,1651 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "house-of-ruqa-backend",
3
+ "version": "1.0.0",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "house-of-ruqa-backend",
9
+ "version": "1.0.0",
10
+ "license": "ISC",
11
+ "dependencies": {
12
+ "cloudinary": "^1.41.3",
13
+ "cors": "^2.8.6",
14
+ "dotenv": "^17.3.1",
15
+ "express": "^5.2.1",
16
+ "mongoose": "^9.3.1",
17
+ "multer": "^2.1.1",
18
+ "multer-storage-cloudinary": "^4.0.0"
19
+ },
20
+ "devDependencies": {
21
+ "nodemon": "^3.1.14"
22
+ }
23
+ },
24
+ "node_modules/@mongodb-js/saslprep": {
25
+ "version": "1.4.6",
26
+ "resolved": "https://registry.npmjs.org/@mongodb-js/saslprep/-/saslprep-1.4.6.tgz",
27
+ "integrity": "sha512-y+x3H1xBZd38n10NZF/rEBlvDOOMQ6LKUTHqr8R9VkJ+mmQOYtJFxIlkkK8fZrtOiL6VixbOBWMbZGBdal3Z1g==",
28
+ "license": "MIT",
29
+ "dependencies": {
30
+ "sparse-bitfield": "^3.0.3"
31
+ }
32
+ },
33
+ "node_modules/@types/webidl-conversions": {
34
+ "version": "7.0.3",
35
+ "resolved": "https://registry.npmjs.org/@types/webidl-conversions/-/webidl-conversions-7.0.3.tgz",
36
+ "integrity": "sha512-CiJJvcRtIgzadHCYXw7dqEnMNRjhGZlYK05Mj9OyktqV8uVT8fD2BFOB7S1uwBE3Kj2Z+4UyPmFw/Ixgw/LAlA==",
37
+ "license": "MIT"
38
+ },
39
+ "node_modules/@types/whatwg-url": {
40
+ "version": "13.0.0",
41
+ "resolved": "https://registry.npmjs.org/@types/whatwg-url/-/whatwg-url-13.0.0.tgz",
42
+ "integrity": "sha512-N8WXpbE6Wgri7KUSvrmQcqrMllKZ9uxkYWMt+mCSGwNc0Hsw9VQTW7ApqI4XNrx6/SaM2QQJCzMPDEXE058s+Q==",
43
+ "license": "MIT",
44
+ "dependencies": {
45
+ "@types/webidl-conversions": "*"
46
+ }
47
+ },
48
+ "node_modules/accepts": {
49
+ "version": "2.0.0",
50
+ "resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz",
51
+ "integrity": "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng==",
52
+ "license": "MIT",
53
+ "dependencies": {
54
+ "mime-types": "^3.0.0",
55
+ "negotiator": "^1.0.0"
56
+ },
57
+ "engines": {
58
+ "node": ">= 0.6"
59
+ }
60
+ },
61
+ "node_modules/anymatch": {
62
+ "version": "3.1.3",
63
+ "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz",
64
+ "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==",
65
+ "dev": true,
66
+ "license": "ISC",
67
+ "dependencies": {
68
+ "normalize-path": "^3.0.0",
69
+ "picomatch": "^2.0.4"
70
+ },
71
+ "engines": {
72
+ "node": ">= 8"
73
+ }
74
+ },
75
+ "node_modules/append-field": {
76
+ "version": "1.0.0",
77
+ "resolved": "https://registry.npmjs.org/append-field/-/append-field-1.0.0.tgz",
78
+ "integrity": "sha512-klpgFSWLW1ZEs8svjfb7g4qWY0YS5imI82dTg+QahUvJ8YqAY0P10Uk8tTyh9ZGuYEZEMaeJYCF5BFuX552hsw==",
79
+ "license": "MIT"
80
+ },
81
+ "node_modules/balanced-match": {
82
+ "version": "4.0.4",
83
+ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz",
84
+ "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==",
85
+ "dev": true,
86
+ "license": "MIT",
87
+ "engines": {
88
+ "node": "18 || 20 || >=22"
89
+ }
90
+ },
91
+ "node_modules/binary-extensions": {
92
+ "version": "2.3.0",
93
+ "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz",
94
+ "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==",
95
+ "dev": true,
96
+ "license": "MIT",
97
+ "engines": {
98
+ "node": ">=8"
99
+ },
100
+ "funding": {
101
+ "url": "https://github.com/sponsors/sindresorhus"
102
+ }
103
+ },
104
+ "node_modules/body-parser": {
105
+ "version": "2.2.2",
106
+ "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-2.2.2.tgz",
107
+ "integrity": "sha512-oP5VkATKlNwcgvxi0vM0p/D3n2C3EReYVX+DNYs5TjZFn/oQt2j+4sVJtSMr18pdRr8wjTcBl6LoV+FUwzPmNA==",
108
+ "license": "MIT",
109
+ "dependencies": {
110
+ "bytes": "^3.1.2",
111
+ "content-type": "^1.0.5",
112
+ "debug": "^4.4.3",
113
+ "http-errors": "^2.0.0",
114
+ "iconv-lite": "^0.7.0",
115
+ "on-finished": "^2.4.1",
116
+ "qs": "^6.14.1",
117
+ "raw-body": "^3.0.1",
118
+ "type-is": "^2.0.1"
119
+ },
120
+ "engines": {
121
+ "node": ">=18"
122
+ },
123
+ "funding": {
124
+ "type": "opencollective",
125
+ "url": "https://opencollective.com/express"
126
+ }
127
+ },
128
+ "node_modules/brace-expansion": {
129
+ "version": "5.0.4",
130
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.4.tgz",
131
+ "integrity": "sha512-h+DEnpVvxmfVefa4jFbCf5HdH5YMDXRsmKflpf1pILZWRFlTbJpxeU55nJl4Smt5HQaGzg1o6RHFPJaOqnmBDg==",
132
+ "dev": true,
133
+ "license": "MIT",
134
+ "dependencies": {
135
+ "balanced-match": "^4.0.2"
136
+ },
137
+ "engines": {
138
+ "node": "18 || 20 || >=22"
139
+ }
140
+ },
141
+ "node_modules/braces": {
142
+ "version": "3.0.3",
143
+ "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz",
144
+ "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==",
145
+ "dev": true,
146
+ "license": "MIT",
147
+ "dependencies": {
148
+ "fill-range": "^7.1.1"
149
+ },
150
+ "engines": {
151
+ "node": ">=8"
152
+ }
153
+ },
154
+ "node_modules/bson": {
155
+ "version": "7.2.0",
156
+ "resolved": "https://registry.npmjs.org/bson/-/bson-7.2.0.tgz",
157
+ "integrity": "sha512-YCEo7KjMlbNlyHhz7zAZNDpIpQbd+wOEHJYezv0nMYTn4x31eIUM2yomNNubclAt63dObUzKHWsBLJ9QcZNSnQ==",
158
+ "license": "Apache-2.0",
159
+ "engines": {
160
+ "node": ">=20.19.0"
161
+ }
162
+ },
163
+ "node_modules/buffer-from": {
164
+ "version": "1.1.2",
165
+ "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz",
166
+ "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==",
167
+ "license": "MIT"
168
+ },
169
+ "node_modules/busboy": {
170
+ "version": "1.6.0",
171
+ "resolved": "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz",
172
+ "integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==",
173
+ "dependencies": {
174
+ "streamsearch": "^1.1.0"
175
+ },
176
+ "engines": {
177
+ "node": ">=10.16.0"
178
+ }
179
+ },
180
+ "node_modules/bytes": {
181
+ "version": "3.1.2",
182
+ "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz",
183
+ "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==",
184
+ "license": "MIT",
185
+ "engines": {
186
+ "node": ">= 0.8"
187
+ }
188
+ },
189
+ "node_modules/call-bind-apply-helpers": {
190
+ "version": "1.0.2",
191
+ "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz",
192
+ "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==",
193
+ "license": "MIT",
194
+ "dependencies": {
195
+ "es-errors": "^1.3.0",
196
+ "function-bind": "^1.1.2"
197
+ },
198
+ "engines": {
199
+ "node": ">= 0.4"
200
+ }
201
+ },
202
+ "node_modules/call-bound": {
203
+ "version": "1.0.4",
204
+ "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz",
205
+ "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==",
206
+ "license": "MIT",
207
+ "dependencies": {
208
+ "call-bind-apply-helpers": "^1.0.2",
209
+ "get-intrinsic": "^1.3.0"
210
+ },
211
+ "engines": {
212
+ "node": ">= 0.4"
213
+ },
214
+ "funding": {
215
+ "url": "https://github.com/sponsors/ljharb"
216
+ }
217
+ },
218
+ "node_modules/chokidar": {
219
+ "version": "3.6.0",
220
+ "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz",
221
+ "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==",
222
+ "dev": true,
223
+ "license": "MIT",
224
+ "dependencies": {
225
+ "anymatch": "~3.1.2",
226
+ "braces": "~3.0.2",
227
+ "glob-parent": "~5.1.2",
228
+ "is-binary-path": "~2.1.0",
229
+ "is-glob": "~4.0.1",
230
+ "normalize-path": "~3.0.0",
231
+ "readdirp": "~3.6.0"
232
+ },
233
+ "engines": {
234
+ "node": ">= 8.10.0"
235
+ },
236
+ "funding": {
237
+ "url": "https://paulmillr.com/funding/"
238
+ },
239
+ "optionalDependencies": {
240
+ "fsevents": "~2.3.2"
241
+ }
242
+ },
243
+ "node_modules/cloudinary": {
244
+ "version": "1.41.3",
245
+ "resolved": "https://registry.npmjs.org/cloudinary/-/cloudinary-1.41.3.tgz",
246
+ "integrity": "sha512-4o84y+E7dbif3lMns+p3UW6w6hLHEifbX/7zBJvaih1E9QNMZITENQ14GPYJC4JmhygYXsuuBb9bRA3xWEoOfg==",
247
+ "license": "MIT",
248
+ "dependencies": {
249
+ "cloudinary-core": "^2.13.0",
250
+ "core-js": "^3.30.1",
251
+ "lodash": "^4.17.21",
252
+ "q": "^1.5.1"
253
+ },
254
+ "engines": {
255
+ "node": ">=0.6"
256
+ }
257
+ },
258
+ "node_modules/cloudinary-core": {
259
+ "version": "2.14.1",
260
+ "resolved": "https://registry.npmjs.org/cloudinary-core/-/cloudinary-core-2.14.1.tgz",
261
+ "integrity": "sha512-57rgZSQD2cJsz1rga6M7jIDQuEAzkwvq63vTvs3/I8rNpGLyHMoKoIvBkNS0Guv5RZ9KDReJhI2LmElk4D9U1g==",
262
+ "license": "MIT",
263
+ "peerDependencies": {
264
+ "lodash": ">=4.0"
265
+ }
266
+ },
267
+ "node_modules/concat-stream": {
268
+ "version": "2.0.0",
269
+ "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-2.0.0.tgz",
270
+ "integrity": "sha512-MWufYdFw53ccGjCA+Ol7XJYpAlW6/prSMzuPOTRnJGcGzuhLn4Scrz7qf6o8bROZ514ltazcIFJZevcfbo0x7A==",
271
+ "engines": [
272
+ "node >= 6.0"
273
+ ],
274
+ "license": "MIT",
275
+ "dependencies": {
276
+ "buffer-from": "^1.0.0",
277
+ "inherits": "^2.0.3",
278
+ "readable-stream": "^3.0.2",
279
+ "typedarray": "^0.0.6"
280
+ }
281
+ },
282
+ "node_modules/content-disposition": {
283
+ "version": "1.0.1",
284
+ "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.0.1.tgz",
285
+ "integrity": "sha512-oIXISMynqSqm241k6kcQ5UwttDILMK4BiurCfGEREw6+X9jkkpEe5T9FZaApyLGGOnFuyMWZpdolTXMtvEJ08Q==",
286
+ "license": "MIT",
287
+ "engines": {
288
+ "node": ">=18"
289
+ },
290
+ "funding": {
291
+ "type": "opencollective",
292
+ "url": "https://opencollective.com/express"
293
+ }
294
+ },
295
+ "node_modules/content-type": {
296
+ "version": "1.0.5",
297
+ "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz",
298
+ "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==",
299
+ "license": "MIT",
300
+ "engines": {
301
+ "node": ">= 0.6"
302
+ }
303
+ },
304
+ "node_modules/cookie": {
305
+ "version": "0.7.2",
306
+ "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz",
307
+ "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==",
308
+ "license": "MIT",
309
+ "engines": {
310
+ "node": ">= 0.6"
311
+ }
312
+ },
313
+ "node_modules/cookie-signature": {
314
+ "version": "1.2.2",
315
+ "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.2.2.tgz",
316
+ "integrity": "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg==",
317
+ "license": "MIT",
318
+ "engines": {
319
+ "node": ">=6.6.0"
320
+ }
321
+ },
322
+ "node_modules/core-js": {
323
+ "version": "3.49.0",
324
+ "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.49.0.tgz",
325
+ "integrity": "sha512-es1U2+YTtzpwkxVLwAFdSpaIMyQaq0PBgm3YD1W3Qpsn1NAmO3KSgZfu+oGSWVu6NvLHoHCV/aYcsE5wiB7ALg==",
326
+ "hasInstallScript": true,
327
+ "license": "MIT",
328
+ "funding": {
329
+ "type": "opencollective",
330
+ "url": "https://opencollective.com/core-js"
331
+ }
332
+ },
333
+ "node_modules/cors": {
334
+ "version": "2.8.6",
335
+ "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.6.tgz",
336
+ "integrity": "sha512-tJtZBBHA6vjIAaF6EnIaq6laBBP9aq/Y3ouVJjEfoHbRBcHBAHYcMh/w8LDrk2PvIMMq8gmopa5D4V8RmbrxGw==",
337
+ "license": "MIT",
338
+ "dependencies": {
339
+ "object-assign": "^4",
340
+ "vary": "^1"
341
+ },
342
+ "engines": {
343
+ "node": ">= 0.10"
344
+ },
345
+ "funding": {
346
+ "type": "opencollective",
347
+ "url": "https://opencollective.com/express"
348
+ }
349
+ },
350
+ "node_modules/debug": {
351
+ "version": "4.4.3",
352
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
353
+ "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==",
354
+ "license": "MIT",
355
+ "dependencies": {
356
+ "ms": "^2.1.3"
357
+ },
358
+ "engines": {
359
+ "node": ">=6.0"
360
+ },
361
+ "peerDependenciesMeta": {
362
+ "supports-color": {
363
+ "optional": true
364
+ }
365
+ }
366
+ },
367
+ "node_modules/depd": {
368
+ "version": "2.0.0",
369
+ "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
370
+ "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
371
+ "license": "MIT",
372
+ "engines": {
373
+ "node": ">= 0.8"
374
+ }
375
+ },
376
+ "node_modules/dotenv": {
377
+ "version": "17.3.1",
378
+ "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-17.3.1.tgz",
379
+ "integrity": "sha512-IO8C/dzEb6O3F9/twg6ZLXz164a2fhTnEWb95H23Dm4OuN+92NmEAlTrupP9VW6Jm3sO26tQlqyvyi4CsnY9GA==",
380
+ "license": "BSD-2-Clause",
381
+ "engines": {
382
+ "node": ">=12"
383
+ },
384
+ "funding": {
385
+ "url": "https://dotenvx.com"
386
+ }
387
+ },
388
+ "node_modules/dunder-proto": {
389
+ "version": "1.0.1",
390
+ "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
391
+ "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==",
392
+ "license": "MIT",
393
+ "dependencies": {
394
+ "call-bind-apply-helpers": "^1.0.1",
395
+ "es-errors": "^1.3.0",
396
+ "gopd": "^1.2.0"
397
+ },
398
+ "engines": {
399
+ "node": ">= 0.4"
400
+ }
401
+ },
402
+ "node_modules/ee-first": {
403
+ "version": "1.1.1",
404
+ "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
405
+ "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==",
406
+ "license": "MIT"
407
+ },
408
+ "node_modules/encodeurl": {
409
+ "version": "2.0.0",
410
+ "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz",
411
+ "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==",
412
+ "license": "MIT",
413
+ "engines": {
414
+ "node": ">= 0.8"
415
+ }
416
+ },
417
+ "node_modules/es-define-property": {
418
+ "version": "1.0.1",
419
+ "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
420
+ "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==",
421
+ "license": "MIT",
422
+ "engines": {
423
+ "node": ">= 0.4"
424
+ }
425
+ },
426
+ "node_modules/es-errors": {
427
+ "version": "1.3.0",
428
+ "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
429
+ "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
430
+ "license": "MIT",
431
+ "engines": {
432
+ "node": ">= 0.4"
433
+ }
434
+ },
435
+ "node_modules/es-object-atoms": {
436
+ "version": "1.1.1",
437
+ "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz",
438
+ "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==",
439
+ "license": "MIT",
440
+ "dependencies": {
441
+ "es-errors": "^1.3.0"
442
+ },
443
+ "engines": {
444
+ "node": ">= 0.4"
445
+ }
446
+ },
447
+ "node_modules/escape-html": {
448
+ "version": "1.0.3",
449
+ "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
450
+ "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==",
451
+ "license": "MIT"
452
+ },
453
+ "node_modules/etag": {
454
+ "version": "1.8.1",
455
+ "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
456
+ "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==",
457
+ "license": "MIT",
458
+ "engines": {
459
+ "node": ">= 0.6"
460
+ }
461
+ },
462
+ "node_modules/express": {
463
+ "version": "5.2.1",
464
+ "resolved": "https://registry.npmjs.org/express/-/express-5.2.1.tgz",
465
+ "integrity": "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw==",
466
+ "license": "MIT",
467
+ "dependencies": {
468
+ "accepts": "^2.0.0",
469
+ "body-parser": "^2.2.1",
470
+ "content-disposition": "^1.0.0",
471
+ "content-type": "^1.0.5",
472
+ "cookie": "^0.7.1",
473
+ "cookie-signature": "^1.2.1",
474
+ "debug": "^4.4.0",
475
+ "depd": "^2.0.0",
476
+ "encodeurl": "^2.0.0",
477
+ "escape-html": "^1.0.3",
478
+ "etag": "^1.8.1",
479
+ "finalhandler": "^2.1.0",
480
+ "fresh": "^2.0.0",
481
+ "http-errors": "^2.0.0",
482
+ "merge-descriptors": "^2.0.0",
483
+ "mime-types": "^3.0.0",
484
+ "on-finished": "^2.4.1",
485
+ "once": "^1.4.0",
486
+ "parseurl": "^1.3.3",
487
+ "proxy-addr": "^2.0.7",
488
+ "qs": "^6.14.0",
489
+ "range-parser": "^1.2.1",
490
+ "router": "^2.2.0",
491
+ "send": "^1.1.0",
492
+ "serve-static": "^2.2.0",
493
+ "statuses": "^2.0.1",
494
+ "type-is": "^2.0.1",
495
+ "vary": "^1.1.2"
496
+ },
497
+ "engines": {
498
+ "node": ">= 18"
499
+ },
500
+ "funding": {
501
+ "type": "opencollective",
502
+ "url": "https://opencollective.com/express"
503
+ }
504
+ },
505
+ "node_modules/fill-range": {
506
+ "version": "7.1.1",
507
+ "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz",
508
+ "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==",
509
+ "dev": true,
510
+ "license": "MIT",
511
+ "dependencies": {
512
+ "to-regex-range": "^5.0.1"
513
+ },
514
+ "engines": {
515
+ "node": ">=8"
516
+ }
517
+ },
518
+ "node_modules/finalhandler": {
519
+ "version": "2.1.1",
520
+ "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-2.1.1.tgz",
521
+ "integrity": "sha512-S8KoZgRZN+a5rNwqTxlZZePjT/4cnm0ROV70LedRHZ0p8u9fRID0hJUZQpkKLzro8LfmC8sx23bY6tVNxv8pQA==",
522
+ "license": "MIT",
523
+ "dependencies": {
524
+ "debug": "^4.4.0",
525
+ "encodeurl": "^2.0.0",
526
+ "escape-html": "^1.0.3",
527
+ "on-finished": "^2.4.1",
528
+ "parseurl": "^1.3.3",
529
+ "statuses": "^2.0.1"
530
+ },
531
+ "engines": {
532
+ "node": ">= 18.0.0"
533
+ },
534
+ "funding": {
535
+ "type": "opencollective",
536
+ "url": "https://opencollective.com/express"
537
+ }
538
+ },
539
+ "node_modules/forwarded": {
540
+ "version": "0.2.0",
541
+ "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
542
+ "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==",
543
+ "license": "MIT",
544
+ "engines": {
545
+ "node": ">= 0.6"
546
+ }
547
+ },
548
+ "node_modules/fresh": {
549
+ "version": "2.0.0",
550
+ "resolved": "https://registry.npmjs.org/fresh/-/fresh-2.0.0.tgz",
551
+ "integrity": "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A==",
552
+ "license": "MIT",
553
+ "engines": {
554
+ "node": ">= 0.8"
555
+ }
556
+ },
557
+ "node_modules/fsevents": {
558
+ "version": "2.3.3",
559
+ "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
560
+ "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
561
+ "dev": true,
562
+ "hasInstallScript": true,
563
+ "license": "MIT",
564
+ "optional": true,
565
+ "os": [
566
+ "darwin"
567
+ ],
568
+ "engines": {
569
+ "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
570
+ }
571
+ },
572
+ "node_modules/function-bind": {
573
+ "version": "1.1.2",
574
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
575
+ "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
576
+ "license": "MIT",
577
+ "funding": {
578
+ "url": "https://github.com/sponsors/ljharb"
579
+ }
580
+ },
581
+ "node_modules/get-intrinsic": {
582
+ "version": "1.3.0",
583
+ "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
584
+ "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==",
585
+ "license": "MIT",
586
+ "dependencies": {
587
+ "call-bind-apply-helpers": "^1.0.2",
588
+ "es-define-property": "^1.0.1",
589
+ "es-errors": "^1.3.0",
590
+ "es-object-atoms": "^1.1.1",
591
+ "function-bind": "^1.1.2",
592
+ "get-proto": "^1.0.1",
593
+ "gopd": "^1.2.0",
594
+ "has-symbols": "^1.1.0",
595
+ "hasown": "^2.0.2",
596
+ "math-intrinsics": "^1.1.0"
597
+ },
598
+ "engines": {
599
+ "node": ">= 0.4"
600
+ },
601
+ "funding": {
602
+ "url": "https://github.com/sponsors/ljharb"
603
+ }
604
+ },
605
+ "node_modules/get-proto": {
606
+ "version": "1.0.1",
607
+ "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
608
+ "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==",
609
+ "license": "MIT",
610
+ "dependencies": {
611
+ "dunder-proto": "^1.0.1",
612
+ "es-object-atoms": "^1.0.0"
613
+ },
614
+ "engines": {
615
+ "node": ">= 0.4"
616
+ }
617
+ },
618
+ "node_modules/glob-parent": {
619
+ "version": "5.1.2",
620
+ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
621
+ "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
622
+ "dev": true,
623
+ "license": "ISC",
624
+ "dependencies": {
625
+ "is-glob": "^4.0.1"
626
+ },
627
+ "engines": {
628
+ "node": ">= 6"
629
+ }
630
+ },
631
+ "node_modules/gopd": {
632
+ "version": "1.2.0",
633
+ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz",
634
+ "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==",
635
+ "license": "MIT",
636
+ "engines": {
637
+ "node": ">= 0.4"
638
+ },
639
+ "funding": {
640
+ "url": "https://github.com/sponsors/ljharb"
641
+ }
642
+ },
643
+ "node_modules/has-flag": {
644
+ "version": "3.0.0",
645
+ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
646
+ "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==",
647
+ "dev": true,
648
+ "license": "MIT",
649
+ "engines": {
650
+ "node": ">=4"
651
+ }
652
+ },
653
+ "node_modules/has-symbols": {
654
+ "version": "1.1.0",
655
+ "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz",
656
+ "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==",
657
+ "license": "MIT",
658
+ "engines": {
659
+ "node": ">= 0.4"
660
+ },
661
+ "funding": {
662
+ "url": "https://github.com/sponsors/ljharb"
663
+ }
664
+ },
665
+ "node_modules/hasown": {
666
+ "version": "2.0.2",
667
+ "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
668
+ "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
669
+ "license": "MIT",
670
+ "dependencies": {
671
+ "function-bind": "^1.1.2"
672
+ },
673
+ "engines": {
674
+ "node": ">= 0.4"
675
+ }
676
+ },
677
+ "node_modules/http-errors": {
678
+ "version": "2.0.1",
679
+ "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz",
680
+ "integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==",
681
+ "license": "MIT",
682
+ "dependencies": {
683
+ "depd": "~2.0.0",
684
+ "inherits": "~2.0.4",
685
+ "setprototypeof": "~1.2.0",
686
+ "statuses": "~2.0.2",
687
+ "toidentifier": "~1.0.1"
688
+ },
689
+ "engines": {
690
+ "node": ">= 0.8"
691
+ },
692
+ "funding": {
693
+ "type": "opencollective",
694
+ "url": "https://opencollective.com/express"
695
+ }
696
+ },
697
+ "node_modules/iconv-lite": {
698
+ "version": "0.7.2",
699
+ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.7.2.tgz",
700
+ "integrity": "sha512-im9DjEDQ55s9fL4EYzOAv0yMqmMBSZp6G0VvFyTMPKWxiSBHUj9NW/qqLmXUwXrrM7AvqSlTCfvqRb0cM8yYqw==",
701
+ "license": "MIT",
702
+ "dependencies": {
703
+ "safer-buffer": ">= 2.1.2 < 3.0.0"
704
+ },
705
+ "engines": {
706
+ "node": ">=0.10.0"
707
+ },
708
+ "funding": {
709
+ "type": "opencollective",
710
+ "url": "https://opencollective.com/express"
711
+ }
712
+ },
713
+ "node_modules/ignore-by-default": {
714
+ "version": "1.0.1",
715
+ "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz",
716
+ "integrity": "sha512-Ius2VYcGNk7T90CppJqcIkS5ooHUZyIQK+ClZfMfMNFEF9VSE73Fq+906u/CWu92x4gzZMWOwfFYckPObzdEbA==",
717
+ "dev": true,
718
+ "license": "ISC"
719
+ },
720
+ "node_modules/inherits": {
721
+ "version": "2.0.4",
722
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
723
+ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
724
+ "license": "ISC"
725
+ },
726
+ "node_modules/ipaddr.js": {
727
+ "version": "1.9.1",
728
+ "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
729
+ "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
730
+ "license": "MIT",
731
+ "engines": {
732
+ "node": ">= 0.10"
733
+ }
734
+ },
735
+ "node_modules/is-binary-path": {
736
+ "version": "2.1.0",
737
+ "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
738
+ "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
739
+ "dev": true,
740
+ "license": "MIT",
741
+ "dependencies": {
742
+ "binary-extensions": "^2.0.0"
743
+ },
744
+ "engines": {
745
+ "node": ">=8"
746
+ }
747
+ },
748
+ "node_modules/is-extglob": {
749
+ "version": "2.1.1",
750
+ "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
751
+ "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
752
+ "dev": true,
753
+ "license": "MIT",
754
+ "engines": {
755
+ "node": ">=0.10.0"
756
+ }
757
+ },
758
+ "node_modules/is-glob": {
759
+ "version": "4.0.3",
760
+ "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
761
+ "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
762
+ "dev": true,
763
+ "license": "MIT",
764
+ "dependencies": {
765
+ "is-extglob": "^2.1.1"
766
+ },
767
+ "engines": {
768
+ "node": ">=0.10.0"
769
+ }
770
+ },
771
+ "node_modules/is-number": {
772
+ "version": "7.0.0",
773
+ "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
774
+ "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
775
+ "dev": true,
776
+ "license": "MIT",
777
+ "engines": {
778
+ "node": ">=0.12.0"
779
+ }
780
+ },
781
+ "node_modules/is-promise": {
782
+ "version": "4.0.0",
783
+ "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz",
784
+ "integrity": "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ==",
785
+ "license": "MIT"
786
+ },
787
+ "node_modules/kareem": {
788
+ "version": "3.2.0",
789
+ "resolved": "https://registry.npmjs.org/kareem/-/kareem-3.2.0.tgz",
790
+ "integrity": "sha512-VS8MWZz/cT+SqBCpVfNN4zoVz5VskR3N4+sTmUXme55e9avQHntpwpNq0yjnosISXqwJ3AQVjlbI4Dyzv//JtA==",
791
+ "license": "Apache-2.0",
792
+ "engines": {
793
+ "node": ">=18.0.0"
794
+ }
795
+ },
796
+ "node_modules/lodash": {
797
+ "version": "4.17.23",
798
+ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.23.tgz",
799
+ "integrity": "sha512-LgVTMpQtIopCi79SJeDiP0TfWi5CNEc/L/aRdTh3yIvmZXTnheWpKjSZhnvMl8iXbC1tFg9gdHHDMLoV7CnG+w==",
800
+ "license": "MIT"
801
+ },
802
+ "node_modules/math-intrinsics": {
803
+ "version": "1.1.0",
804
+ "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
805
+ "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==",
806
+ "license": "MIT",
807
+ "engines": {
808
+ "node": ">= 0.4"
809
+ }
810
+ },
811
+ "node_modules/media-typer": {
812
+ "version": "1.1.0",
813
+ "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-1.1.0.tgz",
814
+ "integrity": "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw==",
815
+ "license": "MIT",
816
+ "engines": {
817
+ "node": ">= 0.8"
818
+ }
819
+ },
820
+ "node_modules/memory-pager": {
821
+ "version": "1.5.0",
822
+ "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz",
823
+ "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==",
824
+ "license": "MIT"
825
+ },
826
+ "node_modules/merge-descriptors": {
827
+ "version": "2.0.0",
828
+ "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-2.0.0.tgz",
829
+ "integrity": "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g==",
830
+ "license": "MIT",
831
+ "engines": {
832
+ "node": ">=18"
833
+ },
834
+ "funding": {
835
+ "url": "https://github.com/sponsors/sindresorhus"
836
+ }
837
+ },
838
+ "node_modules/mime-db": {
839
+ "version": "1.54.0",
840
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.54.0.tgz",
841
+ "integrity": "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ==",
842
+ "license": "MIT",
843
+ "engines": {
844
+ "node": ">= 0.6"
845
+ }
846
+ },
847
+ "node_modules/mime-types": {
848
+ "version": "3.0.2",
849
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-3.0.2.tgz",
850
+ "integrity": "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A==",
851
+ "license": "MIT",
852
+ "dependencies": {
853
+ "mime-db": "^1.54.0"
854
+ },
855
+ "engines": {
856
+ "node": ">=18"
857
+ },
858
+ "funding": {
859
+ "type": "opencollective",
860
+ "url": "https://opencollective.com/express"
861
+ }
862
+ },
863
+ "node_modules/minimatch": {
864
+ "version": "10.2.4",
865
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.4.tgz",
866
+ "integrity": "sha512-oRjTw/97aTBN0RHbYCdtF1MQfvusSIBQM0IZEgzl6426+8jSC0nF1a/GmnVLpfB9yyr6g6FTqWqiZVbxrtaCIg==",
867
+ "dev": true,
868
+ "license": "BlueOak-1.0.0",
869
+ "dependencies": {
870
+ "brace-expansion": "^5.0.2"
871
+ },
872
+ "engines": {
873
+ "node": "18 || 20 || >=22"
874
+ },
875
+ "funding": {
876
+ "url": "https://github.com/sponsors/isaacs"
877
+ }
878
+ },
879
+ "node_modules/mongodb": {
880
+ "version": "7.1.0",
881
+ "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-7.1.0.tgz",
882
+ "integrity": "sha512-kMfnKunbolQYwCIyrkxNJFB4Ypy91pYqua5NargS/f8ODNSJxT03ZU3n1JqL4mCzbSih8tvmMEMLpKTT7x5gCg==",
883
+ "license": "Apache-2.0",
884
+ "dependencies": {
885
+ "@mongodb-js/saslprep": "^1.3.0",
886
+ "bson": "^7.1.1",
887
+ "mongodb-connection-string-url": "^7.0.0"
888
+ },
889
+ "engines": {
890
+ "node": ">=20.19.0"
891
+ },
892
+ "peerDependencies": {
893
+ "@aws-sdk/credential-providers": "^3.806.0",
894
+ "@mongodb-js/zstd": "^7.0.0",
895
+ "gcp-metadata": "^7.0.1",
896
+ "kerberos": "^7.0.0",
897
+ "mongodb-client-encryption": ">=7.0.0 <7.1.0",
898
+ "snappy": "^7.3.2",
899
+ "socks": "^2.8.6"
900
+ },
901
+ "peerDependenciesMeta": {
902
+ "@aws-sdk/credential-providers": {
903
+ "optional": true
904
+ },
905
+ "@mongodb-js/zstd": {
906
+ "optional": true
907
+ },
908
+ "gcp-metadata": {
909
+ "optional": true
910
+ },
911
+ "kerberos": {
912
+ "optional": true
913
+ },
914
+ "mongodb-client-encryption": {
915
+ "optional": true
916
+ },
917
+ "snappy": {
918
+ "optional": true
919
+ },
920
+ "socks": {
921
+ "optional": true
922
+ }
923
+ }
924
+ },
925
+ "node_modules/mongodb-connection-string-url": {
926
+ "version": "7.0.1",
927
+ "resolved": "https://registry.npmjs.org/mongodb-connection-string-url/-/mongodb-connection-string-url-7.0.1.tgz",
928
+ "integrity": "sha512-h0AZ9A7IDVwwHyMxmdMXKy+9oNlF0zFoahHiX3vQ8e3KFcSP3VmsmfvtRSuLPxmyv2vjIDxqty8smTgie/SNRQ==",
929
+ "license": "Apache-2.0",
930
+ "dependencies": {
931
+ "@types/whatwg-url": "^13.0.0",
932
+ "whatwg-url": "^14.1.0"
933
+ },
934
+ "engines": {
935
+ "node": ">=20.19.0"
936
+ }
937
+ },
938
+ "node_modules/mongoose": {
939
+ "version": "9.3.1",
940
+ "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-9.3.1.tgz",
941
+ "integrity": "sha512-58DuQti+LlRS74/UfWN4F3wZsC0Yr1dgTWZ2Wd3/TuSvm6rIdyAjDWbx2xGyuBooqJYdAWotVv4mQgVdivh+3Q==",
942
+ "license": "MIT",
943
+ "dependencies": {
944
+ "kareem": "3.2.0",
945
+ "mongodb": "~7.1",
946
+ "mpath": "0.9.0",
947
+ "mquery": "6.0.0",
948
+ "ms": "2.1.3",
949
+ "sift": "17.1.3"
950
+ },
951
+ "engines": {
952
+ "node": ">=20.19.0"
953
+ },
954
+ "funding": {
955
+ "type": "opencollective",
956
+ "url": "https://opencollective.com/mongoose"
957
+ }
958
+ },
959
+ "node_modules/mpath": {
960
+ "version": "0.9.0",
961
+ "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.9.0.tgz",
962
+ "integrity": "sha512-ikJRQTk8hw5DEoFVxHG1Gn9T/xcjtdnOKIU1JTmGjZZlg9LST2mBLmcX3/ICIbgJydT2GOc15RnNy5mHmzfSew==",
963
+ "license": "MIT",
964
+ "engines": {
965
+ "node": ">=4.0.0"
966
+ }
967
+ },
968
+ "node_modules/mquery": {
969
+ "version": "6.0.0",
970
+ "resolved": "https://registry.npmjs.org/mquery/-/mquery-6.0.0.tgz",
971
+ "integrity": "sha512-b2KQNsmgtkscfeDgkYMcWGn9vZI9YoXh802VDEwE6qc50zxBFQ0Oo8ROkawbPAsXCY1/Z1yp0MagqsZStPWJjw==",
972
+ "license": "MIT",
973
+ "engines": {
974
+ "node": ">=20.19.0"
975
+ }
976
+ },
977
+ "node_modules/ms": {
978
+ "version": "2.1.3",
979
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
980
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
981
+ "license": "MIT"
982
+ },
983
+ "node_modules/multer": {
984
+ "version": "2.1.1",
985
+ "resolved": "https://registry.npmjs.org/multer/-/multer-2.1.1.tgz",
986
+ "integrity": "sha512-mo+QTzKlx8R7E5ylSXxWzGoXoZbOsRMpyitcht8By2KHvMbf3tjwosZ/Mu/XYU6UuJ3VZnODIrak5ZrPiPyB6A==",
987
+ "license": "MIT",
988
+ "dependencies": {
989
+ "append-field": "^1.0.0",
990
+ "busboy": "^1.6.0",
991
+ "concat-stream": "^2.0.0",
992
+ "type-is": "^1.6.18"
993
+ },
994
+ "engines": {
995
+ "node": ">= 10.16.0"
996
+ },
997
+ "funding": {
998
+ "type": "opencollective",
999
+ "url": "https://opencollective.com/express"
1000
+ }
1001
+ },
1002
+ "node_modules/multer-storage-cloudinary": {
1003
+ "version": "4.0.0",
1004
+ "resolved": "https://registry.npmjs.org/multer-storage-cloudinary/-/multer-storage-cloudinary-4.0.0.tgz",
1005
+ "integrity": "sha512-25lm9R6o5dWrHLqLvygNX+kBOxprzpmZdnVKH4+r68WcfCt8XV6xfQaMuAg+kUE5Xmr8mJNA4gE0AcBj9FJyWA==",
1006
+ "license": "MIT",
1007
+ "peerDependencies": {
1008
+ "cloudinary": "^1.21.0"
1009
+ }
1010
+ },
1011
+ "node_modules/multer/node_modules/media-typer": {
1012
+ "version": "0.3.0",
1013
+ "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
1014
+ "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==",
1015
+ "license": "MIT",
1016
+ "engines": {
1017
+ "node": ">= 0.6"
1018
+ }
1019
+ },
1020
+ "node_modules/multer/node_modules/mime-db": {
1021
+ "version": "1.52.0",
1022
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
1023
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
1024
+ "license": "MIT",
1025
+ "engines": {
1026
+ "node": ">= 0.6"
1027
+ }
1028
+ },
1029
+ "node_modules/multer/node_modules/mime-types": {
1030
+ "version": "2.1.35",
1031
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
1032
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
1033
+ "license": "MIT",
1034
+ "dependencies": {
1035
+ "mime-db": "1.52.0"
1036
+ },
1037
+ "engines": {
1038
+ "node": ">= 0.6"
1039
+ }
1040
+ },
1041
+ "node_modules/multer/node_modules/type-is": {
1042
+ "version": "1.6.18",
1043
+ "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
1044
+ "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
1045
+ "license": "MIT",
1046
+ "dependencies": {
1047
+ "media-typer": "0.3.0",
1048
+ "mime-types": "~2.1.24"
1049
+ },
1050
+ "engines": {
1051
+ "node": ">= 0.6"
1052
+ }
1053
+ },
1054
+ "node_modules/negotiator": {
1055
+ "version": "1.0.0",
1056
+ "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-1.0.0.tgz",
1057
+ "integrity": "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg==",
1058
+ "license": "MIT",
1059
+ "engines": {
1060
+ "node": ">= 0.6"
1061
+ }
1062
+ },
1063
+ "node_modules/nodemon": {
1064
+ "version": "3.1.14",
1065
+ "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-3.1.14.tgz",
1066
+ "integrity": "sha512-jakjZi93UtB3jHMWsXL68FXSAosbLfY0In5gtKq3niLSkrWznrVBzXFNOEMJUfc9+Ke7SHWoAZsiMkNP3vq6Jw==",
1067
+ "dev": true,
1068
+ "license": "MIT",
1069
+ "dependencies": {
1070
+ "chokidar": "^3.5.2",
1071
+ "debug": "^4",
1072
+ "ignore-by-default": "^1.0.1",
1073
+ "minimatch": "^10.2.1",
1074
+ "pstree.remy": "^1.1.8",
1075
+ "semver": "^7.5.3",
1076
+ "simple-update-notifier": "^2.0.0",
1077
+ "supports-color": "^5.5.0",
1078
+ "touch": "^3.1.0",
1079
+ "undefsafe": "^2.0.5"
1080
+ },
1081
+ "bin": {
1082
+ "nodemon": "bin/nodemon.js"
1083
+ },
1084
+ "engines": {
1085
+ "node": ">=10"
1086
+ },
1087
+ "funding": {
1088
+ "type": "opencollective",
1089
+ "url": "https://opencollective.com/nodemon"
1090
+ }
1091
+ },
1092
+ "node_modules/normalize-path": {
1093
+ "version": "3.0.0",
1094
+ "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
1095
+ "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
1096
+ "dev": true,
1097
+ "license": "MIT",
1098
+ "engines": {
1099
+ "node": ">=0.10.0"
1100
+ }
1101
+ },
1102
+ "node_modules/object-assign": {
1103
+ "version": "4.1.1",
1104
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
1105
+ "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
1106
+ "license": "MIT",
1107
+ "engines": {
1108
+ "node": ">=0.10.0"
1109
+ }
1110
+ },
1111
+ "node_modules/object-inspect": {
1112
+ "version": "1.13.4",
1113
+ "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz",
1114
+ "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==",
1115
+ "license": "MIT",
1116
+ "engines": {
1117
+ "node": ">= 0.4"
1118
+ },
1119
+ "funding": {
1120
+ "url": "https://github.com/sponsors/ljharb"
1121
+ }
1122
+ },
1123
+ "node_modules/on-finished": {
1124
+ "version": "2.4.1",
1125
+ "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
1126
+ "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
1127
+ "license": "MIT",
1128
+ "dependencies": {
1129
+ "ee-first": "1.1.1"
1130
+ },
1131
+ "engines": {
1132
+ "node": ">= 0.8"
1133
+ }
1134
+ },
1135
+ "node_modules/once": {
1136
+ "version": "1.4.0",
1137
+ "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
1138
+ "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
1139
+ "license": "ISC",
1140
+ "dependencies": {
1141
+ "wrappy": "1"
1142
+ }
1143
+ },
1144
+ "node_modules/parseurl": {
1145
+ "version": "1.3.3",
1146
+ "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
1147
+ "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
1148
+ "license": "MIT",
1149
+ "engines": {
1150
+ "node": ">= 0.8"
1151
+ }
1152
+ },
1153
+ "node_modules/path-to-regexp": {
1154
+ "version": "8.3.0",
1155
+ "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-8.3.0.tgz",
1156
+ "integrity": "sha512-7jdwVIRtsP8MYpdXSwOS0YdD0Du+qOoF/AEPIt88PcCFrZCzx41oxku1jD88hZBwbNUIEfpqvuhjFaMAqMTWnA==",
1157
+ "license": "MIT",
1158
+ "funding": {
1159
+ "type": "opencollective",
1160
+ "url": "https://opencollective.com/express"
1161
+ }
1162
+ },
1163
+ "node_modules/picomatch": {
1164
+ "version": "2.3.1",
1165
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz",
1166
+ "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==",
1167
+ "dev": true,
1168
+ "license": "MIT",
1169
+ "engines": {
1170
+ "node": ">=8.6"
1171
+ },
1172
+ "funding": {
1173
+ "url": "https://github.com/sponsors/jonschlinkert"
1174
+ }
1175
+ },
1176
+ "node_modules/proxy-addr": {
1177
+ "version": "2.0.7",
1178
+ "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
1179
+ "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
1180
+ "license": "MIT",
1181
+ "dependencies": {
1182
+ "forwarded": "0.2.0",
1183
+ "ipaddr.js": "1.9.1"
1184
+ },
1185
+ "engines": {
1186
+ "node": ">= 0.10"
1187
+ }
1188
+ },
1189
+ "node_modules/pstree.remy": {
1190
+ "version": "1.1.8",
1191
+ "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz",
1192
+ "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==",
1193
+ "dev": true,
1194
+ "license": "MIT"
1195
+ },
1196
+ "node_modules/punycode": {
1197
+ "version": "2.3.1",
1198
+ "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
1199
+ "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==",
1200
+ "license": "MIT",
1201
+ "engines": {
1202
+ "node": ">=6"
1203
+ }
1204
+ },
1205
+ "node_modules/q": {
1206
+ "version": "1.5.1",
1207
+ "resolved": "https://registry.npmjs.org/q/-/q-1.5.1.tgz",
1208
+ "integrity": "sha512-kV/CThkXo6xyFEZUugw/+pIOywXcDbFYgSct5cT3gqlbkBE1SJdwy6UQoZvodiWF/ckQLZyDE/Bu1M6gVu5lVw==",
1209
+ "deprecated": "You or someone you depend on is using Q, the JavaScript Promise library that gave JavaScript developers strong feelings about promises. They can almost certainly migrate to the native JavaScript promise now. Thank you literally everyone for joining me in this bet against the odds. Be excellent to each other.\n\n(For a CapTP with native promises, see @endo/eventual-send and @endo/captp)",
1210
+ "license": "MIT",
1211
+ "engines": {
1212
+ "node": ">=0.6.0",
1213
+ "teleport": ">=0.2.0"
1214
+ }
1215
+ },
1216
+ "node_modules/qs": {
1217
+ "version": "6.15.0",
1218
+ "resolved": "https://registry.npmjs.org/qs/-/qs-6.15.0.tgz",
1219
+ "integrity": "sha512-mAZTtNCeetKMH+pSjrb76NAM8V9a05I9aBZOHztWy/UqcJdQYNsf59vrRKWnojAT9Y+GbIvoTBC++CPHqpDBhQ==",
1220
+ "license": "BSD-3-Clause",
1221
+ "dependencies": {
1222
+ "side-channel": "^1.1.0"
1223
+ },
1224
+ "engines": {
1225
+ "node": ">=0.6"
1226
+ },
1227
+ "funding": {
1228
+ "url": "https://github.com/sponsors/ljharb"
1229
+ }
1230
+ },
1231
+ "node_modules/range-parser": {
1232
+ "version": "1.2.1",
1233
+ "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
1234
+ "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
1235
+ "license": "MIT",
1236
+ "engines": {
1237
+ "node": ">= 0.6"
1238
+ }
1239
+ },
1240
+ "node_modules/raw-body": {
1241
+ "version": "3.0.2",
1242
+ "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-3.0.2.tgz",
1243
+ "integrity": "sha512-K5zQjDllxWkf7Z5xJdV0/B0WTNqx6vxG70zJE4N0kBs4LovmEYWJzQGxC9bS9RAKu3bgM40lrd5zoLJ12MQ5BA==",
1244
+ "license": "MIT",
1245
+ "dependencies": {
1246
+ "bytes": "~3.1.2",
1247
+ "http-errors": "~2.0.1",
1248
+ "iconv-lite": "~0.7.0",
1249
+ "unpipe": "~1.0.0"
1250
+ },
1251
+ "engines": {
1252
+ "node": ">= 0.10"
1253
+ }
1254
+ },
1255
+ "node_modules/readable-stream": {
1256
+ "version": "3.6.2",
1257
+ "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz",
1258
+ "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==",
1259
+ "license": "MIT",
1260
+ "dependencies": {
1261
+ "inherits": "^2.0.3",
1262
+ "string_decoder": "^1.1.1",
1263
+ "util-deprecate": "^1.0.1"
1264
+ },
1265
+ "engines": {
1266
+ "node": ">= 6"
1267
+ }
1268
+ },
1269
+ "node_modules/readdirp": {
1270
+ "version": "3.6.0",
1271
+ "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz",
1272
+ "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==",
1273
+ "dev": true,
1274
+ "license": "MIT",
1275
+ "dependencies": {
1276
+ "picomatch": "^2.2.1"
1277
+ },
1278
+ "engines": {
1279
+ "node": ">=8.10.0"
1280
+ }
1281
+ },
1282
+ "node_modules/router": {
1283
+ "version": "2.2.0",
1284
+ "resolved": "https://registry.npmjs.org/router/-/router-2.2.0.tgz",
1285
+ "integrity": "sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ==",
1286
+ "license": "MIT",
1287
+ "dependencies": {
1288
+ "debug": "^4.4.0",
1289
+ "depd": "^2.0.0",
1290
+ "is-promise": "^4.0.0",
1291
+ "parseurl": "^1.3.3",
1292
+ "path-to-regexp": "^8.0.0"
1293
+ },
1294
+ "engines": {
1295
+ "node": ">= 18"
1296
+ }
1297
+ },
1298
+ "node_modules/safe-buffer": {
1299
+ "version": "5.2.1",
1300
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
1301
+ "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
1302
+ "funding": [
1303
+ {
1304
+ "type": "github",
1305
+ "url": "https://github.com/sponsors/feross"
1306
+ },
1307
+ {
1308
+ "type": "patreon",
1309
+ "url": "https://www.patreon.com/feross"
1310
+ },
1311
+ {
1312
+ "type": "consulting",
1313
+ "url": "https://feross.org/support"
1314
+ }
1315
+ ],
1316
+ "license": "MIT"
1317
+ },
1318
+ "node_modules/safer-buffer": {
1319
+ "version": "2.1.2",
1320
+ "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
1321
+ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
1322
+ "license": "MIT"
1323
+ },
1324
+ "node_modules/semver": {
1325
+ "version": "7.7.4",
1326
+ "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.4.tgz",
1327
+ "integrity": "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA==",
1328
+ "dev": true,
1329
+ "license": "ISC",
1330
+ "bin": {
1331
+ "semver": "bin/semver.js"
1332
+ },
1333
+ "engines": {
1334
+ "node": ">=10"
1335
+ }
1336
+ },
1337
+ "node_modules/send": {
1338
+ "version": "1.2.1",
1339
+ "resolved": "https://registry.npmjs.org/send/-/send-1.2.1.tgz",
1340
+ "integrity": "sha512-1gnZf7DFcoIcajTjTwjwuDjzuz4PPcY2StKPlsGAQ1+YH20IRVrBaXSWmdjowTJ6u8Rc01PoYOGHXfP1mYcZNQ==",
1341
+ "license": "MIT",
1342
+ "dependencies": {
1343
+ "debug": "^4.4.3",
1344
+ "encodeurl": "^2.0.0",
1345
+ "escape-html": "^1.0.3",
1346
+ "etag": "^1.8.1",
1347
+ "fresh": "^2.0.0",
1348
+ "http-errors": "^2.0.1",
1349
+ "mime-types": "^3.0.2",
1350
+ "ms": "^2.1.3",
1351
+ "on-finished": "^2.4.1",
1352
+ "range-parser": "^1.2.1",
1353
+ "statuses": "^2.0.2"
1354
+ },
1355
+ "engines": {
1356
+ "node": ">= 18"
1357
+ },
1358
+ "funding": {
1359
+ "type": "opencollective",
1360
+ "url": "https://opencollective.com/express"
1361
+ }
1362
+ },
1363
+ "node_modules/serve-static": {
1364
+ "version": "2.2.1",
1365
+ "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-2.2.1.tgz",
1366
+ "integrity": "sha512-xRXBn0pPqQTVQiC8wyQrKs2MOlX24zQ0POGaj0kultvoOCstBQM5yvOhAVSUwOMjQtTvsPWoNCHfPGwaaQJhTw==",
1367
+ "license": "MIT",
1368
+ "dependencies": {
1369
+ "encodeurl": "^2.0.0",
1370
+ "escape-html": "^1.0.3",
1371
+ "parseurl": "^1.3.3",
1372
+ "send": "^1.2.0"
1373
+ },
1374
+ "engines": {
1375
+ "node": ">= 18"
1376
+ },
1377
+ "funding": {
1378
+ "type": "opencollective",
1379
+ "url": "https://opencollective.com/express"
1380
+ }
1381
+ },
1382
+ "node_modules/setprototypeof": {
1383
+ "version": "1.2.0",
1384
+ "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
1385
+ "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==",
1386
+ "license": "ISC"
1387
+ },
1388
+ "node_modules/side-channel": {
1389
+ "version": "1.1.0",
1390
+ "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz",
1391
+ "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==",
1392
+ "license": "MIT",
1393
+ "dependencies": {
1394
+ "es-errors": "^1.3.0",
1395
+ "object-inspect": "^1.13.3",
1396
+ "side-channel-list": "^1.0.0",
1397
+ "side-channel-map": "^1.0.1",
1398
+ "side-channel-weakmap": "^1.0.2"
1399
+ },
1400
+ "engines": {
1401
+ "node": ">= 0.4"
1402
+ },
1403
+ "funding": {
1404
+ "url": "https://github.com/sponsors/ljharb"
1405
+ }
1406
+ },
1407
+ "node_modules/side-channel-list": {
1408
+ "version": "1.0.0",
1409
+ "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz",
1410
+ "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==",
1411
+ "license": "MIT",
1412
+ "dependencies": {
1413
+ "es-errors": "^1.3.0",
1414
+ "object-inspect": "^1.13.3"
1415
+ },
1416
+ "engines": {
1417
+ "node": ">= 0.4"
1418
+ },
1419
+ "funding": {
1420
+ "url": "https://github.com/sponsors/ljharb"
1421
+ }
1422
+ },
1423
+ "node_modules/side-channel-map": {
1424
+ "version": "1.0.1",
1425
+ "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz",
1426
+ "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==",
1427
+ "license": "MIT",
1428
+ "dependencies": {
1429
+ "call-bound": "^1.0.2",
1430
+ "es-errors": "^1.3.0",
1431
+ "get-intrinsic": "^1.2.5",
1432
+ "object-inspect": "^1.13.3"
1433
+ },
1434
+ "engines": {
1435
+ "node": ">= 0.4"
1436
+ },
1437
+ "funding": {
1438
+ "url": "https://github.com/sponsors/ljharb"
1439
+ }
1440
+ },
1441
+ "node_modules/side-channel-weakmap": {
1442
+ "version": "1.0.2",
1443
+ "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz",
1444
+ "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==",
1445
+ "license": "MIT",
1446
+ "dependencies": {
1447
+ "call-bound": "^1.0.2",
1448
+ "es-errors": "^1.3.0",
1449
+ "get-intrinsic": "^1.2.5",
1450
+ "object-inspect": "^1.13.3",
1451
+ "side-channel-map": "^1.0.1"
1452
+ },
1453
+ "engines": {
1454
+ "node": ">= 0.4"
1455
+ },
1456
+ "funding": {
1457
+ "url": "https://github.com/sponsors/ljharb"
1458
+ }
1459
+ },
1460
+ "node_modules/sift": {
1461
+ "version": "17.1.3",
1462
+ "resolved": "https://registry.npmjs.org/sift/-/sift-17.1.3.tgz",
1463
+ "integrity": "sha512-Rtlj66/b0ICeFzYTuNvX/EF1igRbbnGSvEyT79McoZa/DeGhMyC5pWKOEsZKnpkqtSeovd5FL/bjHWC3CIIvCQ==",
1464
+ "license": "MIT"
1465
+ },
1466
+ "node_modules/simple-update-notifier": {
1467
+ "version": "2.0.0",
1468
+ "resolved": "https://registry.npmjs.org/simple-update-notifier/-/simple-update-notifier-2.0.0.tgz",
1469
+ "integrity": "sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w==",
1470
+ "dev": true,
1471
+ "license": "MIT",
1472
+ "dependencies": {
1473
+ "semver": "^7.5.3"
1474
+ },
1475
+ "engines": {
1476
+ "node": ">=10"
1477
+ }
1478
+ },
1479
+ "node_modules/sparse-bitfield": {
1480
+ "version": "3.0.3",
1481
+ "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz",
1482
+ "integrity": "sha512-kvzhi7vqKTfkh0PZU+2D2PIllw2ymqJKujUcyPMd9Y75Nv4nPbGJZXNhxsgdQab2BmlDct1YnfQCguEvHr7VsQ==",
1483
+ "license": "MIT",
1484
+ "dependencies": {
1485
+ "memory-pager": "^1.0.2"
1486
+ }
1487
+ },
1488
+ "node_modules/statuses": {
1489
+ "version": "2.0.2",
1490
+ "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz",
1491
+ "integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==",
1492
+ "license": "MIT",
1493
+ "engines": {
1494
+ "node": ">= 0.8"
1495
+ }
1496
+ },
1497
+ "node_modules/streamsearch": {
1498
+ "version": "1.1.0",
1499
+ "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-1.1.0.tgz",
1500
+ "integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==",
1501
+ "engines": {
1502
+ "node": ">=10.0.0"
1503
+ }
1504
+ },
1505
+ "node_modules/string_decoder": {
1506
+ "version": "1.3.0",
1507
+ "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz",
1508
+ "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==",
1509
+ "license": "MIT",
1510
+ "dependencies": {
1511
+ "safe-buffer": "~5.2.0"
1512
+ }
1513
+ },
1514
+ "node_modules/supports-color": {
1515
+ "version": "5.5.0",
1516
+ "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
1517
+ "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
1518
+ "dev": true,
1519
+ "license": "MIT",
1520
+ "dependencies": {
1521
+ "has-flag": "^3.0.0"
1522
+ },
1523
+ "engines": {
1524
+ "node": ">=4"
1525
+ }
1526
+ },
1527
+ "node_modules/to-regex-range": {
1528
+ "version": "5.0.1",
1529
+ "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
1530
+ "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
1531
+ "dev": true,
1532
+ "license": "MIT",
1533
+ "dependencies": {
1534
+ "is-number": "^7.0.0"
1535
+ },
1536
+ "engines": {
1537
+ "node": ">=8.0"
1538
+ }
1539
+ },
1540
+ "node_modules/toidentifier": {
1541
+ "version": "1.0.1",
1542
+ "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
1543
+ "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==",
1544
+ "license": "MIT",
1545
+ "engines": {
1546
+ "node": ">=0.6"
1547
+ }
1548
+ },
1549
+ "node_modules/touch": {
1550
+ "version": "3.1.1",
1551
+ "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.1.tgz",
1552
+ "integrity": "sha512-r0eojU4bI8MnHr8c5bNo7lJDdI2qXlWWJk6a9EAFG7vbhTjElYhBVS3/miuE0uOuoLdb8Mc/rVfsmm6eo5o9GA==",
1553
+ "dev": true,
1554
+ "license": "ISC",
1555
+ "bin": {
1556
+ "nodetouch": "bin/nodetouch.js"
1557
+ }
1558
+ },
1559
+ "node_modules/tr46": {
1560
+ "version": "5.1.1",
1561
+ "resolved": "https://registry.npmjs.org/tr46/-/tr46-5.1.1.tgz",
1562
+ "integrity": "sha512-hdF5ZgjTqgAntKkklYw0R03MG2x/bSzTtkxmIRw/sTNV8YXsCJ1tfLAX23lhxhHJlEf3CRCOCGGWw3vI3GaSPw==",
1563
+ "license": "MIT",
1564
+ "dependencies": {
1565
+ "punycode": "^2.3.1"
1566
+ },
1567
+ "engines": {
1568
+ "node": ">=18"
1569
+ }
1570
+ },
1571
+ "node_modules/type-is": {
1572
+ "version": "2.0.1",
1573
+ "resolved": "https://registry.npmjs.org/type-is/-/type-is-2.0.1.tgz",
1574
+ "integrity": "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw==",
1575
+ "license": "MIT",
1576
+ "dependencies": {
1577
+ "content-type": "^1.0.5",
1578
+ "media-typer": "^1.1.0",
1579
+ "mime-types": "^3.0.0"
1580
+ },
1581
+ "engines": {
1582
+ "node": ">= 0.6"
1583
+ }
1584
+ },
1585
+ "node_modules/typedarray": {
1586
+ "version": "0.0.6",
1587
+ "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz",
1588
+ "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==",
1589
+ "license": "MIT"
1590
+ },
1591
+ "node_modules/undefsafe": {
1592
+ "version": "2.0.5",
1593
+ "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.5.tgz",
1594
+ "integrity": "sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA==",
1595
+ "dev": true,
1596
+ "license": "MIT"
1597
+ },
1598
+ "node_modules/unpipe": {
1599
+ "version": "1.0.0",
1600
+ "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
1601
+ "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==",
1602
+ "license": "MIT",
1603
+ "engines": {
1604
+ "node": ">= 0.8"
1605
+ }
1606
+ },
1607
+ "node_modules/util-deprecate": {
1608
+ "version": "1.0.2",
1609
+ "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
1610
+ "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
1611
+ "license": "MIT"
1612
+ },
1613
+ "node_modules/vary": {
1614
+ "version": "1.1.2",
1615
+ "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
1616
+ "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==",
1617
+ "license": "MIT",
1618
+ "engines": {
1619
+ "node": ">= 0.8"
1620
+ }
1621
+ },
1622
+ "node_modules/webidl-conversions": {
1623
+ "version": "7.0.0",
1624
+ "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-7.0.0.tgz",
1625
+ "integrity": "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==",
1626
+ "license": "BSD-2-Clause",
1627
+ "engines": {
1628
+ "node": ">=12"
1629
+ }
1630
+ },
1631
+ "node_modules/whatwg-url": {
1632
+ "version": "14.2.0",
1633
+ "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-14.2.0.tgz",
1634
+ "integrity": "sha512-De72GdQZzNTUBBChsXueQUnPKDkg/5A5zp7pFDuQAj5UFoENpiACU0wlCvzpAGnTkj++ihpKwKyYewn/XNUbKw==",
1635
+ "license": "MIT",
1636
+ "dependencies": {
1637
+ "tr46": "^5.1.0",
1638
+ "webidl-conversions": "^7.0.0"
1639
+ },
1640
+ "engines": {
1641
+ "node": ">=18"
1642
+ }
1643
+ },
1644
+ "node_modules/wrappy": {
1645
+ "version": "1.0.2",
1646
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
1647
+ "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==",
1648
+ "license": "ISC"
1649
+ }
1650
+ }
1651
+ }
package.json ADDED
@@ -0,0 +1,26 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "house-of-ruqa-backend",
3
+ "version": "1.0.0",
4
+ "description": "Backend API for House of Ruqa",
5
+ "main": "server.js",
6
+ "type": "module",
7
+ "scripts": {
8
+ "start": "node server.js",
9
+ "dev": "nodemon server.js"
10
+ },
11
+ "keywords": [],
12
+ "author": "",
13
+ "license": "ISC",
14
+ "dependencies": {
15
+ "cloudinary": "^1.41.3",
16
+ "cors": "^2.8.6",
17
+ "dotenv": "^17.3.1",
18
+ "express": "^5.2.1",
19
+ "mongoose": "^9.3.1",
20
+ "multer": "^2.1.1",
21
+ "multer-storage-cloudinary": "^4.0.0"
22
+ },
23
+ "devDependencies": {
24
+ "nodemon": "^3.1.14"
25
+ }
26
+ }
routes/productRoutes.js ADDED
@@ -0,0 +1,15 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ // routes/productRoutes.js
2
+ import express from 'express';
3
+ import { createProduct, getProducts, getProductById, updateProduct, deleteProduct } from '../controllers/productController.js'; // <-- Added imports
4
+ import { upload } from '../config/cloudinary.js';
5
+
6
+ const router = express.Router();
7
+
8
+ router.get('/', getProducts);
9
+ router.get('/:id', getProductById);
10
+ router.post('/', upload.array('images', 5), createProduct);
11
+
12
+ router.put('/:id', updateProduct);
13
+ router.delete('/:id', deleteProduct);
14
+
15
+ export default router;
routes/subcategoryRoutes.js ADDED
@@ -0,0 +1,13 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import express from 'express';
2
+ import { createSubcategory, getSubcategories, updateSubcategory, deleteSubcategory } from '../controllers/subcategoryController.js';
3
+ import { upload } from '../config/cloudinary.js';
4
+
5
+ const router = express.Router();
6
+
7
+ router.get('/', getSubcategories);
8
+ router.post('/', upload.single('image'), createSubcategory);
9
+
10
+ router.put('/:id', upload.single('image'), updateSubcategory);
11
+ router.delete('/:id', deleteSubcategory);
12
+
13
+ export default router;
server.js ADDED
@@ -0,0 +1,47 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ // server.js
2
+ import express from 'express';
3
+ import dotenv from 'dotenv';
4
+ import cors from 'cors';
5
+ import { connectDB } from './config/db.js';
6
+
7
+ import productRoutes from './routes/productRoutes.js';
8
+ import subcategoryRoutes from './routes/subcategoryRoutes.js';
9
+
10
+ // Load environment variables
11
+ dotenv.config();
12
+
13
+ // Initialize Express App
14
+ const app = express();
15
+
16
+ // Connect to MongoDB
17
+ connectDB();
18
+
19
+ // Middleware
20
+ // CORS allows your React app (running on port 5173) to send data to this server
21
+ app.use(cors({
22
+ origin: [
23
+ 'http://localhost:5173',
24
+ 'https://houseofruqa.pages.dev',
25
+ 'https://houseofruqa.shop'
26
+ ],
27
+ credentials: true
28
+ }));
29
+
30
+ // Allows Express to parse JSON data sent from the frontend
31
+ app.use(express.json());
32
+ app.use(express.urlencoded({ extended: true }));
33
+
34
+ // API Routes
35
+ app.use('/api/products', productRoutes);
36
+ app.use('/api/subcategories', subcategoryRoutes);
37
+
38
+ // Basic Health Check Route
39
+ app.get('/', (req, res) => {
40
+ res.send('House of Ruqa API is running...');
41
+ });
42
+
43
+ // Start the server
44
+ const PORT = process.env.PORT || 5000;
45
+ app.listen(PORT, () => {
46
+ console.log(`🚀 Server running on port ${PORT}`);
47
+ });