Reputation: 793
How do i apply uuid and date so that the filename stored in my database and the filename stored in my S3 bucket are the same?
With this current implementation, the uuid and the date are always the same even if a post was made hours later.
Can someone help, would really appreciate it.
const s3 = new AWS.S3({
accessKeyId: process.env.AWS_ID,
secretAccessKey: process.env.AWS_SECRET,
region:process.env.AWS_REGION
})
const uid =uuidv4();
const date =new Date().toISOString()
const multerS3Config = multerS3({
s3: s3,
bucket: process.env.AWS_BUCKET,
metadata: function (req, file, cb) {
cb(null, { fieldName: file.fieldname }); },
shouldTransform: true,
acl: 'public-read',
contentType: multerS3.AUTO_CONTENT_TYPE,
transforms: [
{
id: 'full',
key: (req, file, cb) => cb(null, file.originalname
+ "-" + `${date}` + "-" + `${uid}` +
"_full.jpg"),
transform: (req, file, cb) => cb(null,
sharp().resize(2000).jpeg({
quality: 50 }))
},
{
id: 'thumb',
key: (req, file, cb) => cb(null,
file.originalname + "-" +
`${date}` + "-" + `${uid}` + "_thumb.jpg"),
transform: (req, file, cb) => cb(null,
sharp().resize(100).jpeg({
quality: 30 }))
},
],
});
const upload = multer({
storage: multerS3Config,
limits: { fieldSize: 25 * 1024 * 1024 },
});
Here is my post request
router.post(
"/",
[ upload.array("images", config.get("maxImageCount"))],
async (req, res) => {
const paths = await req.files.map((file) => ({ originalName: file.originalname + "-" +
`${date}`
+ "-" + `${uid}`}));
await Post.create({
title: req.body.title,
userId: req.body.userId,
Post_Images: paths.map((x) => ({ images: x.originalName })),
},
{
include: [Post_Image] }).then(
res.status(201).send())
With this current implementation the the files are getting stored in both the db and s3.
Also, another question i have is what is the difference between using multer and multer-s3? I tried using multer to post the images to s3 but it did not work so i used multer-s3 and it worked.
UPDATE
const s3 = new AWS.S3({
accessKeyId: process.env.AWS_ID,
secretAccessKey: process.env.AWS_SECRET,
region:process.env.AWS_REGION
})
function getFilename() {
return new Date().toISOString() + '-' + uuidv4();
}
function getTransforms() {
const fileName = getFilename();
return {
transforms:[
{
id: 'full',
key: (req, file, cb) => {
let fname = file.originalname.split(".");
cb(null, fname[0] + '-' + fileName + "_full.jpg")},
transform: (req, file, cb) => cb(null,
sharp().resize(2000).jpeg({
quality: 50
}))
},
{
id: 'thumb',
key: (req, file, cb) => {
let fname = file.originalname.split(".");
cb(null, fname[0] + '-' + fileName + "_thumb.jpg")},
transform: (req, file, cb) => cb(null,
sharp().resize(100).jpeg({
quality: 30
}))
}
],
metadata: (req, file, cb) => {
let fname = file.originalname.split(".");
cb(null, {
fieldName: file.fieldname,
key: fname[0] + '-' + fileName
});
}}}
const multerS3Config = multerS3({
s3: s3,
bucket: process.env.AWS_BUCKET,
shouldTransform: true,
acl: 'public-read',
contentType: multerS3.AUTO_CONTENT_TYPE,
...getTransforms()
});
const upload = multer({
storage: multerS3Config,
limits: { fieldSize: 25 * 1024 * 1024 },
});
Here is my post request
router.post(
"/",
[ upload.array("images", config.get("maxImageCount"))],
async (req, res) => {
const paths = await req.files.map((file) => ({ images:
file.transforms[0].metadata.key}));
await Post.create({
title: req.body.title,
userId: req.body.userId,
Post_Images: paths,
},
{
include: [Post_Image] }).then(
res.status(201).send())
My problem is the date and uuid variables both will initialize when node server start, it never changes until your node server restarts
Upvotes: 1
Views: 895
Reputation: 36144
Quick fixes:
const s3 = new AWS.S3({
accessKeyId: process.env.AWS_ID,
secretAccessKey: process.env.AWS_SECRET,
region: process.env.AWS_REGION
});
date
and uuid
variables both will initialize when node server start, it will never change until your node server restart, you just need to put it in to a function to return every time new file name,function getFilename() {
return new Date().toISOString() + '-' + uuidv4();
}
getFilename()
function, add filename in metadata as well,function getTransforms() {
const fileName = getFilename();
return {
transforms: [
{
id: "full",
key: (req, file, cb) => {
let fname = file.originalname.split(".")
cb(null, fname[0] + '-' + fileName + "_full.jpg")
},
transform: (req, file, cb) => cb(null, sharp().resize(2000).jpeg({ quality: 50 }))
},
{
id: "thumb",
key: (req, file, cb) => {
let fname = file.originalname.split(".")
cb(null, fname[0] + '-' + fileName + "_thumb.jpg")
},
transform: (req, file, cb) => cb(null, sharp().resize(100).jpeg({ quality: 30 }))
}
],
metadata: (req, file, cb) => {
let fname = file.originalname.split(".");
cb(null, {
fieldName: file.fieldname,
key: fname[0] + '-' + fileName + ".jpg"
});
}
}
}
getTransforms()
function, that will return transforms and matadata propertiestransfoems.metadata
that we passed from metadata, this will return only _full name,router.post("/", async (req, res) => {
const multerS3Config = multerS3({
s3: s3,
bucket: process.env.AWS_BUCKET,
shouldTransform: true,
acl: 'public-read',
contentType: multerS3.AUTO_CONTENT_TYPE,
...getTransforms()
});
const upload = multer({
storage: multerS3Config,
limits: { fieldSize: 25 * 1024 * 1024 }
});
upload.array("images", config.get("maxImageCount"))(req, res, async(error) => {
const paths = await req.files.map((file) => ({
images: file.transforms[0].metadata.key
}));
await Post.create({
title: req.body.title,
userId: req.body.userId,
Post_Images: paths,
}, {
include: [Post_Image]
})
});
})
Basically the statement multer-s3 Streaming multer storage engine for AWS S3. is enough to differentiate both multer and multer-s3.
The code has not been tested, you can workaround and see what's happen!
Upvotes: 2