Reputation: 388
I'm using the @aws-sdk/client-s3
and I'm trying to read an image from a private bucket in AWS s3.
In my NextJS code, I have an API that should fetch the image and ultimately set the src
tag of the <img>
in the browser to display it. However I am not able to get it working and i'm unsure of the data I'm receiving back.
Here's the code to retrieve the image:
import { GetObjectCommand, S3Client } from "@aws-sdk/client-s3";
export const getObject = async () => {
const s3Client = new S3Client({
region: process.env.AWS_REGION,
credentials: {
accessKeyId: process.env.AWS_ACCESS_KEY,
secretAccessKey: process.env.AWS_SECRET_KEY,
},
})
console.log('GET OBJECT FUNC')
try {
// Create a helper function to convert a ReadableStream to a string.
const streamToString = (stream) =>
new Promise((resolve, reject) => {
const chunks = [];
stream.on("data", (chunk) => chunks.push(chunk));
stream.on("error", reject);
stream.on("end", () => resolve(Buffer.concat(chunks).toString("utf8")));
});
// Get the object from the Amazon S3 bucket. It is returned as a ReadableStream.
const data = await s3Client.send(new GetObjectCommand({
Bucket: process.env.AWS_BUCKET,
Delimiter: "/",
Key: process.env.AWS_BUCKET_FOLDER + 'Screenshot from 2022-02-09 08-13-26.png'
}));
// return data; // For unit tests.
// Convert the ReadableStream to a string.
const bodyContents = await streamToString(data.Body);
return bodyContents;
} catch (err) {
console.log("Error", err);
}
};
export default async function handler(request, response) {
if (request.method === "GET") {
const data = await getObject()
return response.status(200).json(data);
} else {
return response.status(405).end();
}
}
Its just a screenshot I took and i'm trying to download again. When I console.log
the data returned from the getObject function, It looks a like this:
Then on the frontend side, I have the following function that receives the data
const getImage = async (event) => {
console.log('GET IMAGE')
// console.log(event.target)
let arrayBuffer = await axios.get("/api/get-object", {responseType: 'arraybuffer'})
console.log(arrayBuffer.data)
let base64 = 'data:image/png;base64,' + Buffer.from(arrayBuffer.data).toString('base64')
console.log(base64)
event.target.src = base64
}
And sets the src of the image. But the image isnt showing. The console.log output of the above function is:
GET IMAGE Asset.jsx:18:16
ArrayBuffer { byteLength: 687198 }
Asset.jsx:21:16
…
All those slashes and /ve+ don't look correct to me, but I am at a loss as to what I'm doing wrong. I've tried about 5 different methods from all over stack overflow and I'm not able to get this image to show up. Can anyone help please.
Upvotes: 3
Views: 2968
Reputation: 46
All you have to change is:
...
// Get the object from the Amazon S3 bucket. It is returned as a ReadableStream.
const data = await s3Client.send(new GetObjectCommand({
Bucket: process.env.AWS_BUCKET,
Delimiter: "/",
Key: process.env.AWS_BUCKET_FOLDER + 'Screenshot from 2022-02-09 08-13-26.png'
}));
// return data; // For unit tests.
// Convert the ReadableStream to a string.
const bodyContents = await data.Body.transformToString('base64');
...
Upvotes: 0
Reputation: 458
I think your problem is here:
stream.on("end", () => resolve(Buffer.concat(chunks).toString("utf8")));
and here
let base64 = 'data:image/png;base64,' + Buffer.from(arrayBuffer.data).toString('base64')
You are converting the response string to UTF and then converting to base64 here. Change the AWS stream to base64 like this:
stream.on("end", () => resolve(Buffer.concat(chunks).toString("base64")));
and remove the decode when you use the image here:
let base64 = 'data:image/png;base64,' + arrayBuffer.data;
Upvotes: 3