Cloudflare Docs
Visit R2 on GitHub
Set theme to dark (⇧+D)


R2 implements some extensions on top of the basic S3 API. This page outlines these additional, available features.

​​ Extended metadata using Unicode

The Workers R2 API supports Unicode in keys and values natively without requiring any additional encoding or decoding for the customMetadata field. These fields map to the x-amz-meta--prefixed headers used within the R2 S3-compatible API endpoint.

HTTP header names and values may only contain ASCII characters, which is a small subset of the Unicode character library. To easily accommodate users, R2 adheres to RFC2047 and automatically decodes all x-amz-meta-* header values before storage. On retrieval, any metadata values with unicode are RFC2047-encoded before rendering the response. The length limit for metadata values is applied to the decoded Unicode value.

These headers map to the httpMetadata field in the R2 bindings:

HTTP HeaderProperty Name

If using Unicode in object key names, refer to Unicode Interoperability.

​​ Auto-creating buckets on upload

If you are creating buckets on demand, you might initiate an upload with the assumption that a target bucket exists. In this situation, if you received a NoSuchBucket error, you would probably issue a CreateBucket operation. However, following this approach can cause issues: if the body has already been partially consumed, the upload will need to be aborted. A common solution to this issue, followed by other object storage providers, is to use the HTTP 100 response to detect whether the body should be sent, or if the bucket must be created before retrying the upload. However, Cloudflare does not support the HTTP 100 response. Even if the HTTP 100 response was supported, you would still have additional latency due to the round trips involved.

To support sending an upload with a streaming body to a bucket that may not exist yet, upload operations such as PutObject or CreateMultipartUpload allow you to specify a header that will ensure the NoSuchBucket error is not returned. If the bucket does not exist at the time of upload, it is implicitly instantiated with the following CreateBucket request:

PUT / HTTP/1.1
<CreateBucketConfiguration xmlns="">

This is only useful if you are creating buckets on demand because you do not know the name of the bucket or the preferred access location ahead of time. For example, you have one bucket per one of your customers and the bucket is created on first upload to the bucket and not during account registration. In these cases, the ListBuckets extension, which supports accounts with more than 1,000 buckets, may also be useful.

​​ PutObject

​​ cf-create-bucket-if-missing

Add a cf-create-bucket-if-missing header with the value true to implicitly create the bucket if it does not exist yet. Refer to Auto-creating buckets on upload for a more detailed explanation of when to add this header.

​​ CreateMultipartUpload

​​ cf-create-bucket-if-missing

Add a cf-create-bucket-if-missing header with the value true to implicitly create the bucket if it does not exist yet. Refer to Auto-creating buckets on upload for a detailed explanation of when to add this header.

​​ CopyObject

​​ MERGE metadata directive

The x-amz-metadata-directive allows a MERGE value, in addition to the standard COPY and REPLACE options. When used, MERGE is a combination of COPY and REPLACE, which will COPY any metadata keys from the source object and REPLACE those that are specified in the request with the new value. You cannot use MERGE to remove existing metadata keys from the source — use REPLACE instead.

​​ ListBuckets

ListBuckets supports all the same search parameters as ListObjectsV2 in R2 because some customers may have more than 1,000 buckets. Because tooling, like existing S3 libraries, may not expose a way to set these search parameters, these values may also be sent in via headers. Values in headers take precedence over the search parameters.

Search parameterHTTP HeaderMeaning
prefixcf-prefixShow buckets with this prefix only.
start-aftercf-start-afterShow buckets whose name appears lexicographically in the account.
continuation-tokencf-continuation-tokenResume listing from a previously returned continuation token.
max-keyscf-max-keysReturn this maximum number of buckets. Default and max is 1000.

The XML response contains a NextContinuationToken and IsTruncated elements as appropriate. Since these may not be accessible from existing S3 APIs, these are also available in response headers:

XML Response ElementHTTP Response HeaderMeaning
IsTruncatedcf-is-truncatedThis is set to true if the list of buckets returned is not all the buckets on the account.
NextContinuationTokencf-next-continuation-tokenThis is set to continuation token to pass on a subsequent ListBuckets to resume the listing.
StartAfterThis is the start-after value that was passed in on the request.
KeyCountThe number of buckets returned.
ContinuationTokenThe continuation token that was supplied in the request.
MaxKeysThe max keys that were specified in the request.

​​ PutObject

​​ Conditional operations in PutObject

PutObject supports conditional uploads via the If-Match, If-None-Match, If-Modified-Since, and If-Unmodified-Since headers. These headers behave similarly to how GetObject and HeadObject interpret these headers, rejecting uploads with 304 NotModified or 412 PreconditionFailed error codes when an upload is not necessary or the preceding state does not match, respectively.