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.
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.
{ "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
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
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, }, }); }),]);