api/resources/Products/ProductController.js
2024-08-07 16:54:19 +05:30

831 lines
25 KiB
JavaScript

// import ExcelJS from 'exceljs';
import { Product } from "./ProductModel.js";
import cloudinary from "../../Utils/cloudinary.js";
import { v4 as uuidv4 } from "uuid";
import { CategoryModel } from "../Category/CategoryModel.js";
import { Tax } from "../Tax/tax_model.js";
import XLSX from "xlsx";
import fs from "fs";
import path from "path";
import mongoose from "mongoose";
// Function to handle product upload
export const uploadProducts = async (req, res) => {
try {
if (!req.files || !req.files.file) {
return res.status(400).json({ message: "No file uploaded" });
}
const file = req.files.file;
const filePath = path.join("public", "uploads", file.name);
// Ensure 'uploads' directory exists
if (!fs.existsSync(path.dirname(filePath))) {
fs.mkdirSync(path.dirname(filePath), { recursive: true });
}
// Move the file from temp to the uploads directory
await file.mv(filePath);
// Process the file
const fileBuffer = fs.readFileSync(filePath);
const workbook = XLSX.read(fileBuffer, { type: "buffer" });
const sheetName = workbook.SheetNames[0];
const worksheet = workbook.Sheets[sheetName];
const data = XLSX.utils.sheet_to_json(worksheet, { header: 1 });
if (data.length <= 1) {
return res.status(400).json({ message: "Empty spreadsheet or no data found" });
}
const headers = data[0];
// Map headers from the Excel file to your schema
const headerMapping = {
"Product Name": "name",
"category Name": "category",
price: "price",
"GST Name": "GST",
description: "description",
special_instructions: "special_instructions",
};
const requiredHeaders = Object.keys(headerMapping);
if (!requiredHeaders.every((header) => headers.includes(header))) {
return res.status(400).json({ message: "Missing required columns in spreadsheet" });
}
const errors = [];
const productsProcessed = [];
for (let i = 1; i < data.length; i++) {
const row = data[i];
const item = {};
headers.forEach((header, index) => {
if (headerMapping[header]) {
item[headerMapping[header]] = row[index] !== undefined ? row[index] : "";
}
});
// Initialize error tracking for each item
const missingFields = new Set();
const notFoundErrors = new Set();
let { name, category, price, GST, description, special_instructions } = item;
// Trim leading and trailing spaces from product name and GST
name = name ? name.trim() : "";
GST = GST ? GST.toString().trim() : "";
// Validate required fields
if (!name) missingFields.add("name");
if (!category) missingFields.add("category");
if (price === undefined || price === "") missingFields.add("price");
if (!GST) missingFields.add("GST");
// Validate category
if (category) {
const categoryDoc = await CategoryModel.findOne({
categoryName: { $regex: new RegExp(`^${category.trim()}$`, "i") },
}).exec();
if (categoryDoc) {
item.category = categoryDoc._id;
} else {
notFoundErrors.add("category");
}
} else {
missingFields.add("category");
}
// Validate GST
if (GST) {
const gstDoc = await Tax.findOne({
name: { $regex: new RegExp(`^${GST}$`, "i") },
}).exec();
if (gstDoc) {
item.GST = gstDoc._id;
} else {
notFoundErrors.add("GST");
}
} else {
missingFields.add("GST");
}
// Combine all errors into a single message
let errorMessage = "";
if (missingFields.size > 0) {
errorMessage += `Missing fields: ${Array.from(missingFields).join(", ")}. `;
}
if (notFoundErrors.size > 0) {
errorMessage += `Not found: ${Array.from(notFoundErrors).join(", ")}.`;
}
// If there are errors, push them to the errors array
if (errorMessage.trim()) {
errors.push({
productName: name || "N/A",
category: category || "N/A",
GST: GST || "N/A",
message: errorMessage.trim(),
});
continue;
}
// Ensure fields are set to empty strings if not provided
description = description !== undefined ? description : "";
special_instructions = special_instructions !== undefined ? special_instructions : "";
// Check for existing product
let existingProduct = await Product.findOne({
name: new RegExp(`^${name}$`, "i"),
}).exec();
if (existingProduct) {
// Validate that the existing product can be updated
const updateErrors = [];
if (missingFields.size > 0) {
updateErrors.push(`Missing fields: ${Array.from(missingFields).join(", ")}`);
}
if (notFoundErrors.size > 0) {
updateErrors.push(`Not found: ${Array.from(notFoundErrors).join(", ")}`);
}
if (updateErrors.length > 0) {
errors.push({
productName: name,
message: updateErrors.join(". "),
});
continue;
}
// Update existing product
try {
await Product.updateOne(
{ _id: existingProduct._id },
{
$set: {
category: item.category || existingProduct.category,
price: price !== undefined && price !== "" ? price : existingProduct.price,
GST: item.GST || existingProduct.GST,
description: description, // Ensure description is included
special_instructions: special_instructions, // Ensure special_instructions is included
product_Status: item.product_Status || existingProduct.product_Status || "Active",
},
}
);
productsProcessed.push({ ...existingProduct._doc, ...item }); // Track updated product
} catch (error) {
errors.push({
productName: name,
message: "Failed to update product",
});
}
continue;
}
// Create new product
if (item.category && item.GST) {
const productData = {
name,
category: item.category,
price,
GST: item.GST,
description: description, // Ensure description is included
special_instructions: special_instructions, // Ensure special_instructions is included
product_Status: item.product_Status || "Active",
addedBy: req.user._id,
};
try {
const newProduct = await Product.create(productData);
productsProcessed.push(newProduct); // Track new product
} catch (error) {
errors.push({
productName: name,
message: "Failed to create product",
});
}
}
}
fs.unlinkSync(filePath); // Clean up uploaded file
res.status(201).json({
message:
errors.length > 0
? "Products processed with errors!"
: "Products processed successfully!",
productsProcessed: productsProcessed.length, // Total processed products
errors,
});
} catch (error) {
console.error("Error:", error);
res.status(500).json({ message: error.message || "Something went wrong!" });
}
};
export const createProduct = async (req, res) => {
try {
let findProduct = "";
let product = { _id: "" };
// console.log("req.body", req.body);
if (req.body?.product_id) {
findProduct = await Product.findById(req.body.product_id);
}
const name = req.body?.name;
if (!findProduct) {
const data = await Product.findOne({
name: { $regex: new RegExp(`^${name}$`, "ig") },
}).exec();
if (data)
return res
.status(400)
.json({ message: "Product name already exists!" });
req.body.addedBy = req.user._id;
product = await Product.create(req.body);
} else {
const data = await Product.findOne({
_id: { $ne: findProduct._id },
name: { $regex: new RegExp(`^${name}$`, "ig") },
}).exec();
if (data)
return res
.status(400)
.json({ message: "Product name already exists!" });
product = await Product.findByIdAndUpdate(req.body.product_id, req.body);
}
res.status(201).json({
message: "Product details added successfully!",
product_id: product._id,
});
} catch (error) {
res.status(500).json({
message: error.message ? error.message : "Something went wrong!",
});
}
};
///////////////////////////////////////////////////////////////////////////////////////
export const updateProduct = async (req, res) => {
try {
// console.log("Product ID:", req.params.id);
// console.log("Files:", req.files);
// console.log("Body:", req.body);
const { removedImages } = req.body;
const files = req.files?.images || [];
// Parse removedImages
const removedImageIds = removedImages ? JSON.parse(removedImages) : [];
// Find the product
const product = await Product.findById(req.params.id);
if (!product) {
return res.status(404).json({ message: "Product not found!" });
}
// Existing images
let existingImages = product.image || [];
let newImagesLinks = [];
// Ensure files is always an array
const filesArray = Array.isArray(files) ? files : [files];
// Process new images
for (const file of filesArray) {
if (file && file.tempFilePath) {
// Check if file has tempFilePath
try {
const result = await cloudinary.v2.uploader.upload(
file.tempFilePath,
{
folder: "chemiNova/product",
}
);
newImagesLinks.push({
public_id: result.public_id,
url: result.secure_url,
});
// console.log("Uploaded image:", result.secure_url);
} catch (uploadError) {
console.error("Error uploading image:", uploadError);
return res.status(500).json({
message: "Failed to upload image",
error: uploadError.message,
});
}
}
}
// Remove images from Cloudinary and database
// for (const public_id of removedImageIds) {
// try {
// await cloudinary.v2.uploader.destroy(public_id);
// // console.log("Deleted image from Cloudinary:", public_id);
// } catch (deleteError) {
// console.error("Error deleting image from Cloudinary:", deleteError);
// return res.status(500).json({ message: "Failed to delete image", error: deleteError.message });
// }
// }
// Filter out removed images
const updatedImages = existingImages.filter(
(img) => !removedImageIds.includes(img.public_id)
);
const allImages = [...updatedImages, ...newImagesLinks];
// Update product
product.image = allImages;
await product.save();
res
.status(200)
.json({ message: "Product updated successfully!", images: allImages });
} catch (error) {
console.error(error); // Log error for debugging
res.status(500).json({ message: "Server error!", error: error.message });
}
};
////////////////////////////////////////////////////////////////////////////
//get All Product
export const getAllProductAdmin = async (req, res) => {
try {
const PAGE_SIZE = parseInt(req.query.show) || 10;
const page = parseInt(req.query.page) || 1;
const skip = (page - 1) * PAGE_SIZE;
let filter = {};
if (req.query.name) {
filter.name = {
$regex: new RegExp(req.query.name, "i"),
};
}
if (req.query.category) {
filter.category = mongoose.Types.ObjectId(req.query.category);
}
const total = await Product.countDocuments(filter);
const products = await Product.find(filter)
.populate({
path: "category addedBy GST",
select: "categoryName name tax",
})
.limit(PAGE_SIZE)
.skip(skip)
.sort({ createdAt: -1 })
.exec();
return res.status(200).json({
success: true,
total_data: total,
total_pages: Math.ceil(total / PAGE_SIZE),
products,
});
} catch (error) {
console.error(error); // Add logging for better debugging
res.status(500).json({
success: false,
msg: error.message || "Something went wrong!",
});
}
};
//get All Product User(website)
export const getAllProductUser = async (req, res) => {
try {
const PAGE_SIZE = parseInt(req.query?.show || "10");
const page = parseInt(req.query?.page - 1 || "0");
let obj = {};
if (req.query?.name)
obj.name = {
$regex: new RegExp(req.query.name),
$options: "i",
};
if (req.query?.category) obj.category = req.query.category;
obj.product_Status = "Active";
const total = await Product.countDocuments(obj);
const product = await Product.find(obj)
.populate({
path: "category addedBy GST",
select: "name categoryName tax",
})
.limit(PAGE_SIZE)
.skip(PAGE_SIZE * page)
// .sort("name")
.sort({
createdAt: -1,
})
.exec();
if (product) {
return res.status(200).json({
success: true,
total_data: total,
total_pages: Math.ceil(total / PAGE_SIZE),
product,
});
}
} catch (error) {
res.status(500).json({
success: false,
msg: error.message ? error.message : "Something went wrong!",
});
}
};
//Change Product status
export const ChangeProductStatus = async (req, res) => {
try {
const data = await Product.findById(req.params.id);
if (data) {
if (data?.product_Status === "Active") {
let product = await Product.findByIdAndUpdate(
req.params.id,
{ product_Status: "inActive" },
{ new: true } // Return the updated document
);
return res.status(200).json({
success: true,
msg: "Changed status inActive",
});
} else {
let product = await Product.findByIdAndUpdate(
req.params.id,
{ product_Status: "Active" },
{ new: true } // Return the updated document
);
return res.status(200).json({
success: true,
msg: "Changed status Active",
});
}
}
} catch (error) {
res.status(500).json({
success: false,
msg: error.message ? error.message : "Something went wrong!",
});
}
};
//get One Product
export const getOneProduct = async (req, res) => {
try {
const data = await Product.findById(req.params.id).populate({
path: "category addedBy GST",
select: "name categoryName tax",
});
if (data) {
return res.status(200).json({
success: true,
data,
});
}
} catch (error) {
// console.log(error)
res.status(500).json({
success: false,
msg: error.message ? error.message : "Something went wrong!",
});
}
};
// get all product with device products first
export const getAllProductsDevicesFirst = async (req, res) => {
try {
// we want products with category name Device to be displayed first, so i have first found the products with category name Devices then made another request to find all products and filtered products with category devices , then merged both arrays so we get devices first then all other categories
const categoryName = "Devices";
// Find the category object by name first
const category = await CategoryModel.findOne({ categoryName });
if (!category) {
throw new Error("Category not found");
}
// products with device category
const deviceProducts = await Product.find({
category: category._id,
}).populate("category");
// all products
const allProducts = await Product.find()
.populate({
path: "category gst addedBy",
select: "name categoryName tax",
})
.sort({
createdAt: -1,
});
// filtering out products with device category
const filteredProducts = allProducts.filter((ele) => {
return ele.category?.categoryName !== categoryName;
});
// merging both deviceProcuts and filtered products
const product = deviceProducts.concat(filteredProducts);
if (product) {
return res.status(200).json({
success: true,
product,
});
}
} catch (error) {
res.status(500).json({
success: false,
msg: error.message ? error.message : "Something went wrong!",
});
}
};
// 3.update Product
// export const updateProduct = async (req, res) => {
// const {
// name,
// description,
// price,
// category,
// image,
// gst_amount,
// gst,
// total_amount,
// } = req.body;
// console.log(gst_amount, gst, total_amount);
// try {
// // Prepare an array for the images
// const jsonArray = JSON.parse(image);
// const AllImages = jsonArray.map(({ public_id, url }) => ({
// public_id,
// url,
// }));
// if (req.files && req.files.newImages) {
// const newuploadImages = Array.isArray(req.files.newImages)
// ? req.files.newImages
// : [req.files.newImages];
// const imagesLinks = [];
// for (let i = 0; i < newuploadImages.length; i++) {
// const result = await cloudinary.v2.uploader.upload(
// newuploadImages[i].tempFilePath,
// {
// folder: "chemiNova/product",
// }
// );
// imagesLinks.push({
// public_id: result.public_id,
// url: result.secure_url,
// });
// }
// // Combine the existing images and the newly uploaded images
// const updatedImages = [...AllImages, ...imagesLinks];
// // Perform the product update
// const ModifyProduct = await Product.findOneAndUpdate(
// { _id: req.params.id },
// {
// $set: {
// name,
// description,
// price,
// category,
// image: updatedImages,
// gst,
// gst_amount,
// total_amount,
// },
// },
// { new: true }
// );
// return res.status(200).json({
// success: true,
// ModifyProduct,
// });
// } else {
// const ModifyProduct = await Product.findOneAndUpdate(
// { _id: req.params.id },
// {
// $set: {
// name,
// description,
// price,
// category,
// image: AllImages,
// },
// },
// { new: true }
// );
// return res.status(200).json({
// success: true,
// ModifyProduct,
// });
// }
// } catch (error) {
// res.status(500).json({
// success: false,
// msg: error.message ? error.message : "Something went wrong!",
// });
// }
// };
// export const updateProduct = async (req, res) => {
// const {
// name,
// description,
// price,
// category,
// image,
// gst_amount,
// product_Status,
// gst,
// total_amount,
// } = req.body;
// try {
// // Prepare an array for the images
// const jsonArray = JSON.parse(image);
// const AllImages = jsonArray.map(({ public_id, url }) => ({
// public_id,
// url,
// }));
// let updatedImages = AllImages;
// if (req.files && req.files.newImages) {
// const newUploadImages = Array.isArray(req.files.newImages)
// ? req.files.newImages
// : [req.files.newImages];
// const imagesLinks = [];
// for (let i = 0; i < newUploadImages.length; i++) {
// const result = await cloudinary.v2.uploader.upload(
// newUploadImages[i].tempFilePath,
// {
// folder: "chemiNova/product",
// }
// );
// imagesLinks.push({
// public_id: result.public_id,
// url: result.secure_url,
// });
// }
// // Combine the existing images and the newly uploaded images
// updatedImages = [...AllImages, ...imagesLinks];
// }
// // Perform the product update
// const updatedProduct = await Product.findOneAndUpdate(
// { _id: req.params.id },
// {
// $set: {
// name,
// description,
// product_Status,
// price,
// category,
// image: updatedImages,
// gst,
// gst_amount,
// total_amount,
// },
// },
// { new: true }
// );
// if (!updatedProduct) {
// return res.status(404).json({ success: false, msg: "Product not found" });
// }
// return res.status(200).json({
// success: true,
// updatedProduct,
// });
// } catch (error) {
// console.error("Error updating product:", error);
// res.status(500).json({
// success: false,
// msg: error.message ? error.message : "Something went wrong!",
// });
// }
// };
export const deleteImageFromCloudinary = async (req, res) => {
const { public_id } = req.params;
// console.log("Received public_id:", public_id); // Debugging log
// Ensure public_id is not empty
if (!public_id) {
return res.status(400).json({
success: false,
msg: "Public ID is required!",
});
}
const decodedPublicId = decodeURIComponent(public_id);
try {
const response = await cloudinary.v2.uploader.destroy(decodedPublicId);
if (response.result === "ok") {
return res.status(200).json({
success: true,
msg: "Image Deleted Successfully!",
});
} else {
return res.status(400).json({
success: false,
msg: "Image deletion failed!",
});
}
} catch (error) {
console.error("Error deleting image:", error); // Log error for debugging
return res.status(500).json({
success: false,
msg: error.message || "Something went wrong!",
});
}
};
//delete one Product
export const deleteProduct = async (req, res) => {
try {
if (!req.params.id) {
return res.status(400).json({
success: false,
message: "Please Provide Product ID!",
});
}
const getProduct = await Product.findById(req.params.id);
if (!getProduct) {
return res.status(404).json({
success: false,
message: "Product not Found!",
});
}
// Deleting Images From Cloudinary
if (getProduct?.image?.length > 0) {
for (let i = 0; i < getProduct.image.length; i++) {
await cloudinary.v2.uploader.destroy(getProduct.image[i].public_id);
}
}
//-------------------------//
const product = await Product.findByIdAndDelete(req.params.id);
if (!product) {
return res.status(404).json({ message: "Product Not Found" });
}
await product.remove();
res.status(200).json({
success: true,
message: "Product Deleted Successfully!!",
});
} catch (error) {
res.status(500).json({
success: false,
msg: error.message ? error.message : "Something went wrong!",
});
}
};
export const getProductsByCategory = async (req, res) => {
const { categoryName } = req.params; // Assuming category name is in the route
// console.log(categoryName);
try {
// Find the category object by name first
const category = await CategoryModel.findOne({ categoryName });
if (!category) {
throw new Error("Category not found");
}
const products = await Product.find({ category: category._id }).populate(
"category"
);
// console.log(products);
if (products && products.length > 0) {
return res.status(200).json({
success: true,
products,
});
} else {
return res.status(404).json({
success: false,
msg: "No products found for this category",
});
}
} catch (error) {
res.status(500).json({
success: false,
msg: error.message ? error.message : "Something went wrong!",
});
}
};