mirror of
https://github.com/oven-sh/bun
synced 2026-02-02 15:08:46 +00:00
Add support for Requester Pays in S3 operations (#25514)
- Introduced `requestPayer` option in S3-related functions and structures to handle Requester Pays buckets. - Updated S3 client methods to accept and propagate the `requestPayer` flag. - Enhanced documentation for the `requestPayer` option in the S3 type definitions. - Adjusted existing S3 operations to utilize the `requestPayer` parameter where applicable, ensuring compatibility with AWS S3's Requester Pays feature. - Ensured that the new functionality is integrated into multipart uploads and simple requests. ### What does this PR do? This change allows users to specify whether they are willing to pay for data transfer costs when accessing objects in Requester Pays buckets, improving flexibility and compliance with AWS S3's billing model. This closes #25499 ### How did you verify your code works? I have added a new test file to verify this functionality, and all my tests pass. I also tested this against an actual S3 bucket which can only be accessed if requester pays. I can confirm that it's accessible with `requestPayer` is `true`, and the default of `false` does not allow access. An example bucket is here: s3://hl-mainnet-evm-blocks/0/0/1.rmp.lz4 (my usecase is indexing [hyperliquid block data](https://hyperliquid.gitbook.io/hyperliquid-docs/for-developers/hyperevm/raw-hyperevm-block-data) which is stored in s3, and I want to use bun to index faster) --------- Co-authored-by: Alistair Smith <hi@alistair.sh> Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com> Co-authored-by: Ciro Spaciari <ciro.spaciari@gmail.com>
This commit is contained in:
251
test/js/bun/s3/s3-requester-pays.test.ts
Normal file
251
test/js/bun/s3/s3-requester-pays.test.ts
Normal file
@@ -0,0 +1,251 @@
|
||||
import { S3Client, type S3Options } from "bun";
|
||||
import { describe, expect, it } from "bun:test";
|
||||
|
||||
describe("s3 - Requester Pays", () => {
|
||||
const s3Options: S3Options = {
|
||||
accessKeyId: "test",
|
||||
secretAccessKey: "test",
|
||||
region: "eu-west-3",
|
||||
bucket: "my_bucket",
|
||||
};
|
||||
|
||||
it("should include x-amz-request-payer header when requestPayer is true", async () => {
|
||||
let reqHeaders: Headers | undefined = undefined;
|
||||
using server = Bun.serve({
|
||||
port: 0,
|
||||
async fetch(req) {
|
||||
reqHeaders = req.headers;
|
||||
return new Response("", {
|
||||
headers: {
|
||||
"Content-Type": "text/plain",
|
||||
},
|
||||
status: 200,
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
await S3Client.file("test_file", {
|
||||
...s3Options,
|
||||
endpoint: server.url.href,
|
||||
requestPayer: true,
|
||||
}).write("Test content");
|
||||
|
||||
expect(reqHeaders!.get("authorization")).toInclude("x-amz-request-payer");
|
||||
expect(reqHeaders!.get("x-amz-request-payer")).toBe("requester");
|
||||
});
|
||||
|
||||
it("should NOT include x-amz-request-payer header when requestPayer is false", async () => {
|
||||
let reqHeaders: Headers | undefined = undefined;
|
||||
using server = Bun.serve({
|
||||
port: 0,
|
||||
async fetch(req) {
|
||||
reqHeaders = req.headers;
|
||||
return new Response("", {
|
||||
headers: {
|
||||
"Content-Type": "text/plain",
|
||||
},
|
||||
status: 200,
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
await S3Client.file("test_file", {
|
||||
...s3Options,
|
||||
endpoint: server.url.href,
|
||||
requestPayer: false,
|
||||
}).write("Test content");
|
||||
|
||||
expect(reqHeaders!.get("authorization")).not.toInclude("x-amz-request-payer");
|
||||
expect(reqHeaders!.get("x-amz-request-payer")).toBeNull();
|
||||
});
|
||||
|
||||
it("should NOT include x-amz-request-payer header by default", async () => {
|
||||
let reqHeaders: Headers | undefined = undefined;
|
||||
using server = Bun.serve({
|
||||
port: 0,
|
||||
async fetch(req) {
|
||||
reqHeaders = req.headers;
|
||||
return new Response("", {
|
||||
headers: {
|
||||
"Content-Type": "text/plain",
|
||||
},
|
||||
status: 200,
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
await S3Client.file("test_file", {
|
||||
...s3Options,
|
||||
endpoint: server.url.href,
|
||||
}).write("Test content");
|
||||
|
||||
expect(reqHeaders!.get("authorization")).not.toInclude("x-amz-request-payer");
|
||||
expect(reqHeaders!.get("x-amz-request-payer")).toBeNull();
|
||||
});
|
||||
|
||||
it("should work with S3Client instance", async () => {
|
||||
let reqHeaders: Headers | undefined = undefined;
|
||||
using server = Bun.serve({
|
||||
port: 0,
|
||||
async fetch(req) {
|
||||
reqHeaders = req.headers;
|
||||
return new Response("", {
|
||||
headers: {
|
||||
"Content-Type": "text/plain",
|
||||
},
|
||||
status: 200,
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
const client = new S3Client({
|
||||
...s3Options,
|
||||
endpoint: server.url.href,
|
||||
requestPayer: true,
|
||||
});
|
||||
|
||||
await client.file("test_file").write("Test content");
|
||||
|
||||
expect(reqHeaders!.get("authorization")).toInclude("x-amz-request-payer");
|
||||
expect(reqHeaders!.get("x-amz-request-payer")).toBe("requester");
|
||||
});
|
||||
|
||||
it("should work with file-level options overriding client options", async () => {
|
||||
let reqHeaders: Headers | undefined = undefined;
|
||||
using server = Bun.serve({
|
||||
port: 0,
|
||||
async fetch(req) {
|
||||
reqHeaders = req.headers;
|
||||
return new Response("", {
|
||||
headers: {
|
||||
"Content-Type": "text/plain",
|
||||
},
|
||||
status: 200,
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
// Client has requestPayer: false, but file overrides with true
|
||||
const client = new S3Client({
|
||||
...s3Options,
|
||||
endpoint: server.url.href,
|
||||
requestPayer: false,
|
||||
});
|
||||
|
||||
await client.file("test_file", { requestPayer: true }).write("Test content");
|
||||
|
||||
expect(reqHeaders!.get("authorization")).toInclude("x-amz-request-payer");
|
||||
expect(reqHeaders!.get("x-amz-request-payer")).toBe("requester");
|
||||
});
|
||||
|
||||
it("should include x-amz-request-payer in read operations", async () => {
|
||||
let reqHeaders: Headers | undefined = undefined;
|
||||
const body = "Test content from requester pays bucket";
|
||||
using server = Bun.serve({
|
||||
port: 0,
|
||||
async fetch(req) {
|
||||
reqHeaders = req.headers;
|
||||
return new Response(body, {
|
||||
headers: {
|
||||
"Content-Type": "text/plain",
|
||||
"Content-Length": String(body.length),
|
||||
},
|
||||
status: 200,
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
const file = S3Client.file("test_file", {
|
||||
...s3Options,
|
||||
endpoint: server.url.href,
|
||||
requestPayer: true,
|
||||
});
|
||||
|
||||
await file.text();
|
||||
|
||||
expect(reqHeaders!.get("authorization")).toInclude("x-amz-request-payer");
|
||||
expect(reqHeaders!.get("x-amz-request-payer")).toBe("requester");
|
||||
});
|
||||
|
||||
it("should include x-amz-request-payer in HEAD requests (exists/size/stat)", async () => {
|
||||
let reqHeaders: Headers | undefined = undefined;
|
||||
let reqMethod: string | undefined = undefined;
|
||||
using server = Bun.serve({
|
||||
port: 0,
|
||||
async fetch(req) {
|
||||
reqHeaders = req.headers;
|
||||
reqMethod = req.method;
|
||||
return new Response("", {
|
||||
headers: {
|
||||
"Content-Type": "text/plain",
|
||||
"Content-Length": "100",
|
||||
},
|
||||
status: 200,
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
const file = S3Client.file("test_file", {
|
||||
...s3Options,
|
||||
endpoint: server.url.href,
|
||||
requestPayer: true,
|
||||
});
|
||||
|
||||
await file.exists();
|
||||
|
||||
expect(reqMethod).toBe("HEAD");
|
||||
expect(reqHeaders!.get("authorization")).toInclude("x-amz-request-payer");
|
||||
expect(reqHeaders!.get("x-amz-request-payer")).toBe("requester");
|
||||
});
|
||||
|
||||
it("should include x-amz-request-payer in DELETE requests", async () => {
|
||||
let reqHeaders: Headers | undefined = undefined;
|
||||
let reqMethod: string | undefined = undefined;
|
||||
using server = Bun.serve({
|
||||
port: 0,
|
||||
async fetch(req) {
|
||||
reqHeaders = req.headers;
|
||||
reqMethod = req.method;
|
||||
return new Response("", {
|
||||
status: 204,
|
||||
});
|
||||
},
|
||||
});
|
||||
|
||||
const file = S3Client.file("test_file", {
|
||||
...s3Options,
|
||||
endpoint: server.url.href,
|
||||
requestPayer: true,
|
||||
});
|
||||
|
||||
await file.delete();
|
||||
|
||||
expect(reqMethod).toBe("DELETE");
|
||||
expect(reqHeaders!.get("authorization")).toInclude("x-amz-request-payer");
|
||||
expect(reqHeaders!.get("x-amz-request-payer")).toBe("requester");
|
||||
});
|
||||
|
||||
it("should include x-amz-request-payer in presigned URLs", async () => {
|
||||
const file = S3Client.file("test_file", {
|
||||
...s3Options,
|
||||
requestPayer: true,
|
||||
});
|
||||
|
||||
const presignedUrl = file.presign({ expiresIn: 3600 });
|
||||
const url = new URL(presignedUrl);
|
||||
|
||||
expect(url.searchParams.get("x-amz-request-payer")).toBe("requester");
|
||||
});
|
||||
|
||||
it("should NOT include x-amz-request-payer in presigned URLs when requestPayer is false", async () => {
|
||||
const file = S3Client.file("test_file", {
|
||||
...s3Options,
|
||||
requestPayer: false,
|
||||
});
|
||||
|
||||
const presignedUrl = file.presign({ expiresIn: 3600 });
|
||||
const url = new URL(presignedUrl);
|
||||
|
||||
expect(url.searchParams.get("x-amz-request-payer")).toBeNull();
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user