I'm using @aws-sdk/s3-request-presigner on a Node.js server to generate presigned URLs which my web application is then using to upload files to my S3 bucket. This works well, but I am struggling to add tags to the files.
Here is my current server side code:
const putObjectCommandParams: PutObjectCommandInput = {
Bucket: process.env.AWS_S3_BUCKET_NAME,
Key: fileKey,
ContentType: contentType,
Tagging: "org=abc&y=2021"
};
const command = new PutObjectCommand(putObjectCommandParams);
const signedUrl = await getSignedUrl(s3Client, command, { expiresIn: 3600 });
Here is my current web application code:
const xhr = new XMLHttpRequest();
xhr.open('PUT', signedUploadUrl);
xhr.setRequestHeader('Content-Type', contentType);
xhr.onload = () => {
if (xhr.status === 200) {
alert('File uploaded successfully.');
this.fileUploaded(fileName, presignedResponse.file);
} else {
alert('Could not upload file. Status: ' + xhr.status);
}
};
xhr.onerror = (err) => {
alert('Could not upload file. ' + err);
};
xhr.send(file);
The file uploads successfully and appears in S3, but the tags are blank when I check in the AWS console.
As some other answers suggest, I've tried to add the exact same tag string to an "X-Amz-Tagging" header in the file upload request, but then the upload fails and I then get the following error:
<Error>
<Code>AccessDenied</Code>
<Message>There were headers present in the request which were not signed</Message>
<HeadersNotSigned>x-amz-tagging</HeadersNotSigned>
...
Does anyone know why this might be or can you see anything obvious I am doing wrong? Thank you!
x-amz-tagging
also needs to be signed & sent in the request header, unlike other headers that can be a query string.Taking a look at the source code,
@aws-sdk/s3-request-presigner
will by default 'hoist' all headers - includingx-amz-tagging
- to the query parameters of the presigned URL. An exception isx-amz-server-side-encryption
butx-amz-tagging
should also be added.As a result of the hoisting, S3 doesn't add the tags to the object.
With the current code, and its accompanying SDK implementation, you'll get something similar to:
Note that
x-amz-tagging
is not in the value forX-Amz-SignedHeaders
.Also, only adding it manually to the upload request won't work, as the original signing request didn't include the header to be signed. This is why S3 returns the 'There were headers present in the request which were not signed' error. In other words, 'You're sending the
x-amz-tagging
header when using the URL but you didn't originally sign it when you created the URL'.There are 2 changes that need to be made:
s3-request-presigner
to not hoistx-amz-tagging
, when creating the URL.x-amz-tagging
header provided.Tagging
) as the value for thex-amz-tagging
header when using the URL (to upload).Change 1 can be done by adding
x-amz-tagging
to theunhoistableHeaders
field of the parameter object we pass togetSingedUrl
:Your pre-signed URLs will then start to look like:
Note that
x-amz-tagging
is now within the value forX-Amz-SignedHeaders
, and that it is no longer sent as a query parameter via&x-amz-tagging
.Change 2 can be done by including the
x-amz-tagging
header when making the XHR request:With these changes, the
x-amz-tagging
header will be signed in the URL & also included in the headers of the request, enabling S3 to add the tags to the object.Here is a complete yet minimal working Node.js CLI app to demonstrate the above:
Output: