|
@@ -1,22 +1,24 @@
|
|
|
/** logic for image routes */
|
|
|
import multer from 'multer';
|
|
|
import Image from "../models/image.model.js";
|
|
|
-import fs from 'fs';
|
|
|
+import { v2 as cloudinary } from 'cloudinary';
|
|
|
+import dotenv from 'dotenv';
|
|
|
+import path from 'path';
|
|
|
+import DatauriParser from 'datauri/parser.js';
|
|
|
|
|
|
-const uploadsFolder = '/uploads/';
|
|
|
+dotenv.config();
|
|
|
+const dUri = new DatauriParser();
|
|
|
+const dataUri = req => dUri.format(path.extname(req.file.originalname).toString(), req.file.buffer);
|
|
|
|
|
|
-const storage = multer.diskStorage({
|
|
|
- destination: function (req, file, cb) {
|
|
|
- console.log('multer params', req.params);
|
|
|
- let uploadPath = '.' + uploadsFolder;
|
|
|
- if (req.params.folder) uploadPath += req.params.folder;
|
|
|
- cb(null, uploadPath);
|
|
|
- },
|
|
|
- filename: function (req, file, cb) {
|
|
|
- cb(null, new Date().getTime() + '-' + file.originalname);
|
|
|
- }
|
|
|
+cloudinary.config({
|
|
|
+ cloud_name: process.env.CLOUDINARY_CLOUD_NAME,
|
|
|
+ api_key: process.env.CLOUDINARY_API_KEY,
|
|
|
+ api_secret: process.env.CLOUDINARY_API_SECRET,
|
|
|
});
|
|
|
|
|
|
+const storage = multer.memoryStorage();
|
|
|
+
|
|
|
+/*
|
|
|
const imageFileFilter = (req, file, cb) => {
|
|
|
console.log("File in filter", file);
|
|
|
if (file.mimetype === 'image/jpeg' || file.mimetype === 'image/jpg' || file.mimetype === 'image/png') {
|
|
@@ -24,7 +26,7 @@ const imageFileFilter = (req, file, cb) => {
|
|
|
} else {
|
|
|
cb(null, false);
|
|
|
}
|
|
|
-}
|
|
|
+}*/
|
|
|
|
|
|
export const upload = multer({
|
|
|
storage: storage,
|
|
@@ -32,47 +34,105 @@ export const upload = multer({
|
|
|
// Allow upload files under 10MB
|
|
|
fileSize: 10485760,
|
|
|
},
|
|
|
- fileFilter: imageFileFilter,
|
|
|
+ // fileFilter: imageFileFilter,
|
|
|
});
|
|
|
|
|
|
export const uploadSingleImage = async (req, res) => {
|
|
|
- const image = req.file;
|
|
|
- console.log('images file', image);
|
|
|
- let imagePath = uploadsFolder;
|
|
|
- if (req.params.folder) imagePath += req.params.folder + '/';
|
|
|
- imagePath += image.filename;
|
|
|
- const newImage = new Image({
|
|
|
- categoryName: req.body.category,
|
|
|
- categoryId: req.body.categoryId,
|
|
|
- name: image.filename,
|
|
|
- path: imagePath,
|
|
|
+ const image = dataUri(req).content;
|
|
|
+ const { category, categoryId, tags, name } = req.body;
|
|
|
+
|
|
|
+ const folderName = category && categoryId ? category + '/' + categoryId : req.params.folder;
|
|
|
+ const uploadOptions = {
|
|
|
+ folder: folderName,
|
|
|
+ tags: tags || [],
|
|
|
+ };
|
|
|
+ if (category === 'userProfile') {
|
|
|
+ uploadOptions.folder = category;
|
|
|
+ uploadOptions.public_id = categoryId;
|
|
|
+ uploadOptions.overwrite = true;
|
|
|
+ uploadOptions.transformation = { aspect_ratio: 1, gravity: "faces", crop: "fill" };
|
|
|
+ }
|
|
|
+ cloudinary.uploader.upload(image, uploadOptions).then((cloudinaryImage) => {
|
|
|
+ console.log('upload successful');
|
|
|
+ const { public_id, secure_url, url } = cloudinaryImage;
|
|
|
+ const newImage = new Image({
|
|
|
+ categoryName: category,
|
|
|
+ categoryId,
|
|
|
+ url: secure_url || url,
|
|
|
+ cloudinaryPublicId: public_id,
|
|
|
+ name,
|
|
|
+ });
|
|
|
+ try {
|
|
|
+ newImage.save().then(() => {
|
|
|
+ console.log('added image to ' + folderName, public_id, name);
|
|
|
+ res.status(201).json({ 'message': 'successfully added new Image', 'Image': newImage })
|
|
|
+ });
|
|
|
+ } catch (error) {
|
|
|
+ res.status(409).json({ message: error.message, errorDetails: 'upload successful, but database storage failed' });
|
|
|
+ }
|
|
|
+ }).catch((error) => {
|
|
|
+ res.status(409).json({ message: error.message, errorDetails: 'upload failed' });
|
|
|
});
|
|
|
- try {
|
|
|
- newImage.save().then(() => {
|
|
|
- console.log('added image ' + imagePath);
|
|
|
- res.status(201).json({ 'message': 'successfully added new Image', 'Image': newImage })
|
|
|
+}
|
|
|
+
|
|
|
+async function copySingleImage(image, category, newId) {
|
|
|
+ let returnImage = null;
|
|
|
+ if (image.url) {
|
|
|
+ await cloudinary.uploader.upload(image.url, { folder: category + '/' + newId }).then(async (cloudinaryImage) => {
|
|
|
+ console.log('upload successful');
|
|
|
+ const { public_id, secure_url, url } = cloudinaryImage;
|
|
|
+ const newImage = new Image({
|
|
|
+ categoryName: category,
|
|
|
+ categoryId: newId,
|
|
|
+ url: secure_url || url,
|
|
|
+ cloudinaryPublicId: public_id,
|
|
|
+ name: image.name,
|
|
|
+ });
|
|
|
+ await newImage.save().then((savedImage) => {
|
|
|
+ returnImage = savedImage;
|
|
|
+ }).catch((error) => {
|
|
|
+ console.log('upload of copied image successful, but database storage failed', 'error:', error.message);
|
|
|
+ });
|
|
|
+ }).catch((error) => {
|
|
|
+ console.log('image upload failed because ', error);
|
|
|
});
|
|
|
- } catch (error) {
|
|
|
- res.status(409).json({ message: error.message });
|
|
|
}
|
|
|
+ return returnImage;
|
|
|
+}
|
|
|
+
|
|
|
+export const copyImagesForCategory = async (req, res) => {
|
|
|
+ let { category, oldId, newId } = req.params;
|
|
|
+
|
|
|
+ let newImages = [];
|
|
|
+
|
|
|
+ await Image.find({ categoryName: category, categoryId: oldId }, async function (err, foundImages) {
|
|
|
+ if (err) {
|
|
|
+ console.log('error in find', err);
|
|
|
+ } else {
|
|
|
+ newImages = await Promise.all(foundImages.map(async (i) => {
|
|
|
+ return await copySingleImage(i, 'mealImages', newId)
|
|
|
+ }));
|
|
|
+ if (newImages.length > 0) {
|
|
|
+ res.status(201).json({ 'message': 'successfully copied Images', newImages });
|
|
|
+ } else {
|
|
|
+ res.status(400).json({ 'info': `copying images failed. Check log for more info` });
|
|
|
+ }
|
|
|
+ }
|
|
|
+ });
|
|
|
}
|
|
|
|
|
|
export const deleteSingleImage = async (req, res) => {
|
|
|
- let id = req.params.id;
|
|
|
- console.log('req body', req.body.path);
|
|
|
- let path = '.' + req.body.path;
|
|
|
- Image.findByIdAndDelete(id, {}, function (err, deletionResult) {
|
|
|
+ const image = req.body;
|
|
|
+ Image.findByIdAndDelete(image._id, {}, function (err, deletionResult) {
|
|
|
if (err) {
|
|
|
- res.status(400).json({ 'info': `Deletion of image ${id} failed`, 'message': err.message });
|
|
|
+ res.status(400).json({ 'info': `Deletion of image ${image._id} failed`, 'message': err.message });
|
|
|
} else {
|
|
|
- fs.unlink(path, (err) => {
|
|
|
- if (err) {
|
|
|
- console.log('fs deletion failed', err);
|
|
|
- } else {
|
|
|
- console.log('successfully deleted', path);
|
|
|
- }
|
|
|
+ cloudinary.uploader.destroy(image.cloudinaryPublicId).then((result) => {
|
|
|
+ res.status(201).json({ 'info': 'image deleted', deletionResult, cloudinaryResult: result });
|
|
|
+ }).catch(error => {
|
|
|
+ console.log('image deleted from database but not from cloudinary: ', image, 'reason', error);
|
|
|
+ res.status(201).json({ 'info': 'image deleted from database but not from cloudinary', 'cloudinaryError': error });
|
|
|
});
|
|
|
- res.status(201).json({ 'info': 'image deleted, id: ' + id, deletionResult })
|
|
|
}
|
|
|
});
|
|
|
}
|
|
@@ -80,27 +140,26 @@ export const deleteSingleImage = async (req, res) => {
|
|
|
export const deleteAllImagesFromCategory = async (req, res) => {
|
|
|
let category = req.params.category;
|
|
|
let id = req.params.id;
|
|
|
- Image.find({ categoryName: category, categoryId: id }, function (err, foundImages) {
|
|
|
+ console.log('delete all images from ', category, id);
|
|
|
+ await Image.find({ categoryName: category, categoryId: id }, function (err, foundImages) {
|
|
|
if (err) {
|
|
|
console.log('error in find', err);
|
|
|
} else {
|
|
|
foundImages.forEach(i => {
|
|
|
- fs.unlink('.' + i.path, function (err) {
|
|
|
- if (err) {
|
|
|
- console.log('fs deletion failed', err);
|
|
|
- } else {
|
|
|
- console.log('successfully deleted', i.path);
|
|
|
- }
|
|
|
- })
|
|
|
+ cloudinary.uploader.destroy(i.cloudinaryPublicId).then((result) => {
|
|
|
+ console.log(i.name + ' deleted from cloudinary');
|
|
|
+ }).catch(error => {
|
|
|
+ console.log('deletion of ' + i.name + ' from cloudinary failed because', error);
|
|
|
+ });
|
|
|
});
|
|
|
}
|
|
|
- }).then(() => {
|
|
|
- Image.deleteMany({ categoryName: category, categoryId: id }, {}, function (err, deletionResult) {
|
|
|
- if (err) {
|
|
|
- res.status(400).json({ 'info': `Deletion of images from ${category} ${id} failed`, 'message': err.message });
|
|
|
- } else {
|
|
|
- res.status(201).json({ 'info': `images from ${category} with ${id} deleted`, deletionResult })
|
|
|
- }
|
|
|
- });
|
|
|
+ })
|
|
|
+ await cloudinary.api.delete_folder(category + '/' + id);
|
|
|
+ Image.deleteMany({ categoryName: category, categoryId: id }, {}, function (err, deletionResult) {
|
|
|
+ if (err) {
|
|
|
+ res.status(400).json({ 'info': `Deletion of images from ${category} ${id} failed`, 'message': err.message });
|
|
|
+ } else {
|
|
|
+ res.status(201).json({ 'info': `images from ${category} with ${id} deleted`, deletionResult })
|
|
|
+ }
|
|
|
});
|
|
|
}
|