Skip to content
4th April 2024: This is a preview, whilst production-ready, it means some APIs might change

Storage

Cloudflare R2 is an object storage solution that’s S3 compatible, global, scalable, and can be used to store files, images, videos, and more! It integrates natively with Cloudflare workers, and therefore, with Redwood. It is available locally during development, and is automatically configured when you deploy to Cloudflare.

Setup

To use R2 in your project, you need to create a R2 bucket, and bind it to your worker.

Terminal window
npx wrangler r2 bucket create my-bucket
Creating bucket 'my-bucket'...
✅ Created bucket 'my-bucket' with default storage class of Standard.
Configure your Worker to write objects to this bucket:
{
"r2_buckets": [
{
"bucket_name": "my-bucket",
"binding": "R2",
},
],
}

This will create a bucket called my-bucket, which you’ll have to bind to your worker, which you do by pasting the above into your wrangler.jsonc file.

wrangler.jsonc
{
"r2_buckets": [
{
"bucket_name": "my-bucket",
"binding": "R2",
},
],
}

This will make the my-bucket bucket available via the env.R2 binding in your worker. You can then use this binding to upload, download, and manage files stored in R2 using the standard R2 API.

Usage

RedwoodSDK uses the standard Request/Response objects. When uploading files, the data is streamed directly from the client to R2 storage. Similarly, when downloading files, they are streamed directly from R2 to the client. This streaming approach means files are processed in small chunks rather than loading the entire file into memory, making it memory-efficient and suitable for handling large files.

Uploading Files

src/worker.tsx
import { defineApp } from "@redwoodjs/sdk/worker";
import { route } from "@redwoodjs/sdk/router";
defineApp([
route("/upload/", async ({ request, env }) => {
const formData = await request.formData();
const file = formData.get("file") as File;
// Stream the file directly to R2
const r2ObjectKey = `/storage/${file.name}`;
await env.R2.put(r2ObjectKey, file.stream(), {
httpMetadata: {
contentType: file.type,
},
});
return new Response(JSON.stringify({ key: r2ObjectKey }), {
status: 200,
headers: {
"Content-Type": "application/json",
},
});
}),
]);

Downloading Files

src/worker.tsx
import { defineApp } from "@redwoodjs/sdk/worker";
import { route } from "@redwoodjs/sdk/router";
defineApp([
route("/download/*", async ({ request, params, env }) => {
const object = await env.R2.get("/storage/" + params.$0);
if (object === null) {
return new Response("Object Not Found", { status: 404 });
}
return new Response(object.body, {
headers: {
"Content-Type": object.httpMetadata?.contentType as string,
},
});
}),
]);