This is a lightweight S3 client for Deno and other modern JavaScript runtimes. It is designed to offer all the key
features you may need, with no dependencies. It does not use any Deno-specific features, so it should work with any
runtime that supports the fetch
API, web streams API, and ES modules (ESM).
This client is 100% MIT licensed, and is derived from the excellent MinIO JavaScript Client.
Supported functionality:
for await (const object of client.listObjects(options)) { ... }
client.listObjectsGrouped(...)
)client.exists("key")
client.statObject("key")
client.statObject("key", { headers: { 'x-amz-checksum-mode': 'ENABLED' } })
client.getObject("key", options)
Response
object, so for large files, you can opt to consume the data as a stream
(use the .body
property).client.getPartialObject("key", options)
getObject
, this also supports streaming the response if you want to.client.putObject("key", streamOrData, options)
string
, Uint8Array
, or ReadableStream
client.copyObject({ sourceKey: "source", options }, "dest", options)
client.deleteObject("key")
client.presignedGetObject("key", options)
or
client.getPresignedUrl(method, "key", options)
client.presignedPostObject("key", options)
for direct browser uploadsclient.bucketExists("bucketName")
client.makeBucket("bucketName")
client.removeBucket("bucketName")
deno add @bradenmacdonald/s3-lite-client
import { S3Client } from "jsr:@bradenmacdonald/s3-lite-client@0.9.0";
npx jsr add @bradenmacdonald/s3-lite-client
yarn dlx jsr add @bradenmacdonald/s3-lite-client
pnpm dlx jsr add @bradenmacdonald/s3-lite-client
bunx jsr add @bradenmacdonald/s3-lite-client
<script type="module"> import { S3Client } from "https://esm.sh/jsr/@bradenmacdonald/s3-lite-client@0.9.0"; // Or: const { S3Client } = await import("https://esm.sh/jsr/@bradenmacdonald/s3-lite-client@0.9.0"); </script>
Note: if you're using Node.js, this only works on Node 19+.
List data files from a public data set on Amazon S3:
import { S3Client } from "@bradenmacdonald/s3-lite-client"; const s3client = new S3Client({ endPoint: "https://s3.us-east-1.amazonaws.com", region: "us-east-1", bucket: "openalex", }); // Log data about each object found under the 'data/concepts/' prefix: for await (const obj of s3client.listObjects({ prefix: "data/concepts/" })) { console.log(obj); } // { // type: "Object", // key: "data/concepts/updated_date=2024-01-25/part_000.gz", // etag: "2c9b2843c8d2e9057656e1af1c2a92ad", // size: 44105, // lastModified: 2024-01-25T22:57:43.000Z // }, // ... // Or, to get all the keys (paths) as an array: const keys = await Array.fromAsync(s3client.listObjects(), (entry) => entry.key); // keys = [ // "data/authors/manifest", // "data/authors/updated_date=2023-06-08/part_000.gz", // ... // ]
Uploading and downloading a file using a local MinIO server:
import { S3Client } from "@bradenmacdonald/s3-lite-client"; // Connecting to a local MinIO server: const s3client = new S3Client({ endPoint: "http://localhost:9000", region: "dev-region", bucket: "dev-bucket", accessKey: "AKIA_DEV", secretKey: "secretkey", }); // Upload a file: await s3client.putObject("test.txt", "This is the contents of the file."); // Now download it const result = await s3client.getObject("test.txt"); // and stream the results to a local file: const localOutFile = await Deno.open("test-out.txt", { write: true, createNew: true }); await result.body!.pipeTo(localOutFile.writable); // or instead of streaming, you can consume the whole file into memory by awaiting // result.text(), result.blob(), result.arrayBuffer(), or result.json()
Creating a bucket on the S3 service of a local supabase development server:
const client = new S3Client({ endPoint: "http://127.0.0.1:54321/storage/v1/s3", region: "local", accessKey: "paste from output of supabase start", secretKey: "paste from output of supabase start", }); await client.makeBucket("my-bucket");
Set ACLs, Content-Type, custom metadata, etc. during upload:
await s3client.putObject("key", streamOrData, { metadata: { "x-amz-acl": "public-read", "x-amz-meta-custom": "value", }, });
Create a presigned POST policy for direct uploads from a browser:
// Create a presigned POST policy const { url, fields } = await s3client.presignedPostObject("my-file.txt", { expirySeconds: 3600, // URL expires in 1 hour fields: { "Content-Type": "text/plain", }, }); // In the browser, use the policy for direct uploads: const formData = new FormData(); // Add all required fields from the presigned POST Object.entries(fields).forEach(([key, value]) => { formData.append(key, value); }); // Add the file content formData.append("file", fileInput.files[0]); // Upload the object using the presigned POST const response = await fetch(url, { method: "POST", body: formData, }); if (response.ok) { console.log("File uploaded successfully!"); }
For more examples, check out the tests in integration.ts
To run the tests, please use:
deno lint && deno test
To format the code, use:
deno fmt
To run the integration tests, first start MinIO with this command:
docker run --rm -e MINIO_ROOT_USER=AKIA_DEV -e MINIO_ROOT_PASSWORD=secretkey -e MINIO_REGION_NAME=dev-region -p 9000:9000 -p 9001:9001 --entrypoint /bin/sh minio/minio:RELEASE.2025-02-28T09-55-16Z -c 'mkdir -p /data/dev-bucket && minio server --console-address ":9001" /data'
Then while MinIO is running, run
deno test --allow-net integration.ts
(If you encounter issues and need to debug what MinIO is seeing, run these two commands:)
mc alias set localdebug http://localhost:9000 AKIA_DEV secretkey mc admin trace --verbose --all localdebug
Add Package
deno add jsr:@bradenmacdonald/s3-lite-client
Import symbol
import * as s__lite_client from "@bradenmacdonald/s3-lite-client";
Import directly with a jsr specifier
import * as s__lite_client from "jsr:@bradenmacdonald/s3-lite-client";
Add Package
pnpm i jsr:@bradenmacdonald/s3-lite-client
pnpm dlx jsr add @bradenmacdonald/s3-lite-client
Import symbol
import * as s__lite_client from "@bradenmacdonald/s3-lite-client";
Add Package
yarn add jsr:@bradenmacdonald/s3-lite-client
yarn dlx jsr add @bradenmacdonald/s3-lite-client
Import symbol
import * as s__lite_client from "@bradenmacdonald/s3-lite-client";
Add Package
npx jsr add @bradenmacdonald/s3-lite-client
Import symbol
import * as s__lite_client from "@bradenmacdonald/s3-lite-client";
Add Package
bunx jsr add @bradenmacdonald/s3-lite-client
Import symbol
import * as s__lite_client from "@bradenmacdonald/s3-lite-client";