Simplify your stack and build anything. Or everything.
Build tomorrow’s web with a modern solution you truly own.
Code-based nature means you can build on top of it to power anything.
It’s time to take back your content infrastructure.

Uploads

Shows an Upload enabled collection in the Payload Admin Panel
Admin Panel screenshot depicting a Media Collection with Upload enabled

Here are some common use cases of Uploads:

  • Creating a "Media Library" that contains images for use throughout your site or app
  • Building a Gated Content library where users need to sign up to gain access to downloadable assets like ebook PDFs, whitepapers, etc.
  • Storing publicly available, downloadable assets like software, ZIP files, MP4s, etc.

By simply enabling Upload functionality on a Collection, Payload will automatically transform your Collection into a robust file management / storage solution. The following modifications will be made:

  1. filename, mimeType, and filesize fields will be automatically added to your Collection. Optionally, if you pass imageSizes to your Collection's Upload config, a sizes array will also be added containing auto-resized image sizes and filenames.
  2. The Admin Panel will modify its built-in List component to show a thumbnail for each upload within the List View
  3. The Admin Panel will modify its Edit view(s) to add a new set of corresponding Upload UI which will allow for file upload
  4. The create, update, and delete Collection operations will be modified to support file upload, re-upload, and deletion

Enabling Uploads

Every Payload Collection can opt-in to supporting Uploads by specifying the upload property on the Collection's config to either true or to an object containing upload options.

1
import type { CollectionConfig } from 'payload'
2
3
export const Media: CollectionConfig = {
4
slug: 'media',
5
upload: {
6
staticDir: 'media',
7
imageSizes: [
8
{
9
name: 'thumbnail',
10
width: 400,
11
height: 300,
12
position: 'centre',
13
},
14
{
15
name: 'card',
16
width: 768,
17
height: 1024,
18
position: 'centre',
19
},
20
{
21
name: 'tablet',
22
width: 1024,
23
// By specifying `undefined` or leaving a height undefined,
24
// the image will be sized to a certain width,
25
// but it will retain its original aspect ratio
26
// and calculate a height automatically.
27
height: undefined,
28
position: 'centre',
29
},
30
],
31
adminThumbnail: 'thumbnail',
32
mimeTypes: ['image/*'],
33
},
34
fields: [
35
{
36
name: 'alt',
37
type: 'text',
38
},
39
],
40
}

Collection Upload Options

An asterisk denotes that an option is required.

Option

Description

adminThumbnail

Set the way that the Admin Panel will display thumbnails for this Collection. More

bulkUpload

Allow users to upload in bulk from the list view, default is true

cacheTags

Set to false to disable the cache tag set in the UI for the admin thumbnail component. Useful for when CDNs don't allow certain cache queries.

crop

Set to false to disable the cropping tool in the Admin Panel. Crop is enabled by default. More

disableLocalStorage

Completely disable uploading files to disk locally. More

displayPreview

Enable displaying preview of the uploaded file in Upload fields related to this Collection. Can be locally overridden by displayPreview option in Upload field. More.

externalFileHeaderFilter

Accepts existing headers and returns the headers after filtering or modifying.

filesRequiredOnCreate

Mandate file data on creation, default is true.

filenameCompoundIndex

Field slugs to use for a compound index instead of the default filename index.

focalPoint

Set to false to disable the focal point selection tool in the Admin Panel. The focal point selector is only available when imageSizes or resizeOptions are defined. More

formatOptions

An object with format and options that are used with the Sharp image library to format the upload file. More

handlers

Array of Request handlers to execute when fetching a file, if a handler returns a Response it will be sent to the client. Otherwise Payload will retrieve and send back the file.

imageSizes

If specified, image uploads will be automatically resized in accordance to these image sizes. More

mimeTypes

Restrict mimeTypes in the file picker. Array of valid mimetypes or mimetype wildcards More

pasteURL

Controls whether files can be uploaded from remote URLs by pasting them into the Upload field. Enabled by default. Accepts false to disable or an object with an allowList of valid remote URLs. More

resizeOptions

An object passed to the the Sharp image library to resize the uploaded file. More

staticDir

The folder directory to use to store media in. Can be either an absolute path or relative to the directory that contains your config. Defaults to your collection slug

trimOptions

An object passed to the the Sharp image library to trim the uploaded file. More

withMetadata

If specified, appends metadata to the output image file. Accepts a boolean or a function that receives metadata and req, returning a boolean.

Payload-wide Upload Options

Upload options are specifiable on a Collection by Collection basis, you can also control app wide options by passing your base Payload Config an upload property containing an object supportive of all Busboy configuration options. Click here for more documentation about what you can control.

A common example of what you might want to customize within Payload-wide Upload options would be to increase the allowed fileSize of uploads sent to Payload:

1
import { buildConfig } from 'payload'
2
3
export default buildConfig({
4
collections: [
5
{
6
slug: 'media',
7
fields: [
8
{
9
name: 'alt',
10
type: 'text',
11
},
12
],
13
upload: true,
14
},
15
],
16
upload: {
17
limits: {
18
fileSize: 5000000, // 5MB, written in bytes
19
},
20
},
21
})

Custom filename via hooks

You can customize the filename before it's uploaded to the server by using a beforeOperation hook.

1
beforeOperation: [
2
({ req, operation }) => {
3
if ((operation === 'create' || operation === 'update') && req.file) {
4
req.file.name = 'test.jpg'
5
}
6
},
7
],

The req.file object will have additional information about the file, such as mimeType and extension, and you also have full access to the file data itself. The filename from here will also be threaded to image sizes if they're enabled.

Image Sizes

If you specify an array of imageSizes to your upload config, Payload will automatically crop and resize your uploads to fit each of the sizes specified by your config.

The Admin Panel will also automatically display all available files, including width, height, and file size, for each of your uploaded files.

Behind the scenes, Payload relies on sharp to perform its image resizing. You can specify additional options for sharp to use while resizing your images.

Accessing the resized images in hooks

All auto-resized images are exposed to be re-used in hooks and similar via an object that is bound to req.payloadUploadSizes.

The object will have keys for each size generated, and each key will be set equal to a buffer containing the file data.

Handling Image Enlargement

When an uploaded image is smaller than the defined image size, we have 3 options:

withoutEnlargement: undefined | false | true

  1. undefined [default]: uploading images with smaller width AND height than the image size will return null
  2. false: always enlarge images to the image size
  3. true: if the image is smaller than the image size, return the original image

Custom file name per size

Each image size supports a generateImageName function that can be used to generate a custom file name for the resized image. This function receives the original file name, the resize name, the extension, height and width as arguments.

1
{
2
name: 'thumbnail',
3
width: 400,
4
height: 300,
5
generateImageName: ({ height, sizeName, extension, width }) => {
6
return `custom-${sizeName}-${height}-${width}.${extension}`
7
},
8
}

Crop and Focal Point Selector

This feature is only available for image file types.

Setting crop: false and focalPoint: false in your Upload config will be disable the respective selector in the Admin Panel.

Image cropping occurs before any resizing, the resized images will therefore be generated from the cropped image (not the original image).

If no resizing options are specified (imageSizes or resizeOptions), the focal point selector will not be displayed.

Disabling Local Upload Storage

If you are using a plugin to send your files off to a third-party file storage host or CDN, like Amazon S3 or similar, you may not want to store your files locally at all. You can prevent Payload from writing files to disk by specifying disableLocalStorage: true on your collection's upload config.

Admin Thumbnails

You can specify how Payload retrieves admin thumbnails for your upload-enabled Collections with one of the following:

  1. adminThumbnail as a string, equal to one of your provided image size names.
1
import type { CollectionConfig } from 'payload'
2
3
export const Media: CollectionConfig = {
4
slug: 'media',
5
upload: {
6
adminThumbnail: 'small',
7
imageSizes: [
8
{
9
name: 'small',
10
fit: 'cover',
11
height: 300,
12
width: 900,
13
},
14
{
15
name: 'large',
16
fit: 'cover',
17
height: 600,
18
width: 1800,
19
}
20
]
21
}
22
}
  1. adminThumbnail as a function that takes the document's data and sends back a full URL to load the thumbnail.
1
import type { CollectionConfig } from 'payload'
2
3
export const Media: CollectionConfig = {
4
slug: 'media',
5
upload: {
6
adminThumbnail: ({ doc }) => `https://google.com/custom-path-to-file/${doc.filename}`,
7
}
8
}

MimeTypes

Specifying the mimeTypes property can restrict what files are allowed from the user's file picker. This accepts an array of strings, which can be any valid mimetype or mimetype wildcards

Some example values are: image/*, audio/*, video/*, image/png, application/pdf

Example mimeTypes usage:

1
import type { CollectionConfig } from 'payload'
2
3
export const Media: CollectionConfig = {
4
slug: 'media',
5
upload: {
6
mimeTypes: ['image/*', 'application/pdf'],
7
},
8
}

Uploading Files

To upload a file, use your collection's create endpoint. Send it all the data that your Collection requires, as well as a file key containing the file that you'd like to upload.

Send your request as a multipart/form-data request, using FormData if possible.

1
const fileInput = document.querySelector('#your-file-input') ;
2
const formData = new FormData();
3
4
formData.append('file', fileInput.files[0]);
5
6
fetch('api/:upload-slug', {
7
method: 'POST',
8
body: formData,
9
/**
10
* Do not manually add the Content-Type Header
11
* the browser will handle this.
12
*
13
* headers: {
14
* 'Content-Type': 'multipart/form-data'
15
* }
16
*/
17
})

Uploading Files from Remote URLs

The pasteURL option allows users to fetch files from remote URLs by pasting them into an Upload field. This option is enabled by default and can be configured to either allow unrestricted client-side fetching or restrict server-side fetching to specific trusted domains.

By default, Payload uses client-side fetching, where the browser downloads the file directly from the provided URL. However, client-side fetching will fail if the URL’s server has CORS restrictions, making it suitable only for internal URLs or public URLs without CORS blocks.

To fetch files from restricted URLs that would otherwise be blocked by CORS, use server-side fetching by configuring the pasteURL option with an allowList of trusted domains. This method ensures that Payload downloads the file on the server and streams it to the browser. However, for security reasons, only URLs that match the specified allowList will be allowed.

Configuration Example

Here’s how to configure the pasteURL option to control remote URL fetching:

1
import type { CollectionConfig } from 'payload'
2
3
export const Media: CollectionConfig = {
4
slug: 'media',
5
upload: {
6
pasteURL: {
7
allowList: [
8
{
9
hostname: 'payloadcms.com', // required
10
pathname: '',
11
port: '',
12
protocol: 'https',
13
search: ''
14
},
15
{
16
hostname: 'example.com',
17
pathname: '/images/*',
18
},
19
],
20
},
21
},
22
}
Accepted Values for pasteURL

Option

Description

undefined

Default behavior. Enables client-side fetching for internal or public URLs.

false

Disables the ability to paste URLs into Upload fields.

allowList

Enables server-side fetching for specific trusted URLs. Requires an array of objects defining trusted domains. See the table below for details on AllowItem.

AllowItem Properties

An asterisk denotes that an option is required.

Option

Description

Example

hostname *

The hostname of the allowed URL. This is required to ensure the URL is coming from a trusted source.

example.com

pathname

The path portion of the URL. Supports wildcards to match multiple paths.

/images/*

port

The port number of the URL. If not specified, the default port for the protocol will be used.

3000

protocol

The protocol to match. Must be either http or https. Defaults to https.

https

search

The query string of the URL. If specified, the URL must match this exact query string.

?version=1

Access Control

All files that are uploaded to each Collection automatically support the read Access Control function from the Collection itself. You can use this to control who should be allowed to see your uploads, and who should not.

Next

Storage Adapters