This package provides a simple way to use Hetzner Object Storage with Payload CMS.
- Store media files in Hetzner Object Storage instead of local disk
- Support for client-side direct uploads to bypass server upload limits
- Full support for Payload's image resizing
- Compatible with Payload's access control for non-public files
- Built on the AWS SDK to interface with Hetzner's S3-compatible API
npm install @joneslloyd/payload-storage-hetzner
# or with yarn
yarn add @joneslloyd/payload-storage-hetzner
# or with pnpm
pnpm add @joneslloyd/payload-storage-hetzner
import { buildConfig } from 'payload'
import { hetznerStorage } from '@joneslloyd/payload-storage-hetzner'
export default buildConfig({
collections: [
// Your collections that use uploads
{
slug: 'media',
upload: {
// Payload upload configuration
},
},
],
plugins: [
hetznerStorage({
collections: {
media: true, // Enable for the 'media' collection
// Or with prefix
'media-with-prefix': {
prefix: 'custom-folder', // Files will be stored in 'custom-folder/'
},
},
bucket: process.env.HETZNER_BUCKET,
region: 'fsn1', // 'fsn1', 'nbg1', or 'hel1'
credentials: {
accessKeyId: process.env.HETZNER_ACCESS_KEY_ID,
secretAccessKey: process.env.HETZNER_SECRET_ACCESS_KEY,
},
// Optional: enable client-side uploads to bypass server limits
clientUploads: true,
// Optional: set ACL for uploaded files
acl: 'public-read',
// Optional: set Cache-Control header for uploaded files
cacheControl: 'max-age=31536000', // 1 year cache
}),
],
})
Option | Type | Description |
---|---|---|
bucket * |
string |
The name of your Hetzner Object Storage bucket |
region * |
'fsn1' | 'nbg1' | 'hel1' |
The region of your bucket (Falkenstein, Nuremberg, or Helsinki) |
credentials * |
{ accessKeyId: string, secretAccessKey: string } |
Your Hetzner Object Storage credentials |
collections * |
Record<string, CollectionOptions | true> |
Object with keys matching collection slugs where you want to enable storage |
acl |
'private' | 'public-read' |
Access control list for uploads. Default: none |
cacheControl |
string |
Cache-Control header value for uploaded files. Default: none |
clientUploads |
boolean | object |
Enable client-side uploads. Default: false |
disableLocalStorage |
boolean |
If files should not be stored locally. Default: true |
enabled |
boolean |
Whether to enable this plugin. Default: true |
* Required options
By default, this plugin maintains Payload's access control. Your file URLs will remain the same (/:collectionSlug/file/:filename
), and Payload will apply its access control policies when files are requested.
If you want to disable this behavior and use direct URLs to Hetzner Object Storage, you can set disablePayloadAccessControl: true
in the collection options:
hetznerStorage({
collections: {
media: {
disablePayloadAccessControl: true,
},
},
// other options...
})
When disabling Payload's access control, make sure to set your bucket visibility in Hetzner to public
if you want files to be publicly accessible.
You can also set Cache-Control headers on uploaded files to improve performance:
hetznerStorage({
// ...
cacheControl: 'max-age=31536000', // Cache for 1 year
})
Common values: max-age=31536000
(1 year), max-age=86400
(1 day), no-cache
(always revalidate). This applies to both server-side and client-side uploads.
To allow larger file uploads that might exceed server limits (especially on serverless platforms), you can enable client-side uploads directly to Hetzner Object Storage:
hetznerStorage({
// ...
clientUploads: true,
})
When enabling client uploads, make sure to configure CORS on your Hetzner Object Storage bucket to allow PUT requests from your domain:
- Install the AWS CLI and configure it with your Hetzner credentials
- Create a CORS configuration file (
cors.json
):
{
"CORSRules": [
{
"AllowedOrigins": ["https://your-domain.com"],
"AllowedHeaders": ["*"],
"AllowedMethods": ["GET", "PUT", "POST", "DELETE", "HEAD"],
"MaxAgeSeconds": 3000
}
]
}
- Apply the CORS configuration:
aws s3api put-bucket-cors --bucket your-bucket-name --cors-configuration file://cors.json --endpoint-url https://your-region.your-objectstorage.com
Hetzner currently doesn't support custom domain names for buckets directly. If you want to use a custom domain, you'll need to set up domain forwarding. See the Hetzner documentation for more information.
This adapter is built on Hetzner's S3-compatible API. Hetzner supports most but not all S3 features. Notable limitations:
- No support for custom domains for buckets
- Limited encryption support (only SSE-C)
- No support for some S3 features like request-payment, notifications, website hosting, etc.
Refer to the Hetzner documentation for a full list of supported actions.
MIT