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.

Search Plugin

NPM

This plugin generates records of your documents that are extremely fast to search on. It does so by creating a new search collection that is indexed in the database then saving a static copy of each of your documents using only search-critical data. Search records are automatically created, synced, and deleted behind-the-scenes as you manage your application's documents.

For example, if you have a posts collection that is extremely large and complex, this would allow you to sync just the title, excerpt, and slug of each post so you can query on that instead of the original post directly. Search records are static, so querying them also has the significant advantage of bypassing any hooks that may present be on the original documents. You define exactly what data is synced, and you can even modify or fallback this data before it is saved on a per-document basis.

To query search results, use all the existing Payload APIs that you are already familiar with. You can also prioritize search results by setting a custom priority for each collection. For example, you may want to list blog posts before pages. Or you may want one specific post to always take appear first. Search records are given a priority field that can be used as the ?sort= parameter in your queries.

This plugin is a great way to implement a fast, immersive search experience such as a search bar in a front-end application. Many applications may not need the power and complexity of a third-party service like Algolia or ElasticSearch. This plugin provides a first-party alternative that is easy to set up and runs entirely on your own database.

Core Features

  • Automatically adds an indexed search collection to your database
  • Automatically creates, syncs, and deletes search records as you manage your documents
  • Saves only search-critical data that you define (e.g. title, excerpt, etc.)
  • Allows you to query search results using first-party Payload APIs
  • Allows you to query documents without triggering any of their underlying hooks
  • Allows you to easily prioritize search results by collection or document

Installation

Install the plugin using any JavaScript package manager like Yarn, NPM, or PNPM:

1
pnpm add @payloadcms/plugin-search

Basic Usage

In the plugins array of your Payload Config, call the plugin with options:

1
import { buildConfig } from 'payload'
2
import { searchPlugin } from '@payloadcms/plugin-search'
3
4
const config = buildConfig({
5
collections: [
6
{
7
slug: 'pages',
8
fields: [],
9
},
10
{
11
slug: 'posts',
12
fields: [],
13
},
14
],
15
plugins: [
16
searchPlugin({
17
collections: ['pages', 'posts'],
18
defaultPriorities: {
19
pages: 10,
20
posts: 20,
21
},
22
}),
23
],
24
})
25
26
export default config

Options

collections

The collections property is an array of collection slugs to enable syncing to search. Enabled collections receive a beforeChange and afterDelete hook that creates, updates, and deletes its respective search record as it changes over time.

localize

By default, the search plugin will add localization: true to the title field of the newly added search collection if you have localization enabled. If you would like to disable this behavior, you can set this to false.

defaultPriorities

This plugin automatically adds a priority field to the search collection that can be used as the ?sort= parameter in your queries. For example, you may want to list blog posts before pages. Or you may want one specific post to always take appear first.

The defaultPriorities property is used to set a fallback priority on search records during the create operation. It accepts an object with keys that are your collection slugs and values that can either be a number or a function that returns a number. The function receives the doc as an argument, which is the document being created.

1
// payload.config.ts
2
{
3
// ...
4
searchPlugin({
5
defaultPriorities: {
6
pages: ({ doc }) => (doc.title.startsWith('Hello, world!') ? 1 : 10),
7
posts: 20,
8
},
9
}),
10
}

searchOverrides

This plugin automatically creates the search collection, but you can override anything on this collection via the searchOverrides property. It accepts anything from the Payload Collection Config and merges it in with the default search collection config provided by the plugin.

Note that the fields property is a function that receives an object with a defaultFields key. This is an array of fields that are automatically added to the search collection. You can modify this array or add new fields to it.

1
// payload.config.ts
2
{
3
// ...
4
searchPlugin({
5
searchOverrides: {
6
slug: 'search-results',
7
fields: ({ defaultFields }) => [
8
...defaultFields,
9
{
10
name: 'excerpt',
11
type: 'textarea',
12
admin: {
13
position: 'sidebar',
14
},
15
},
16
],
17
},
18
}),
19
}

beforeSync

Before creating or updating a search record, the beforeSync function runs. This is an afterChange hook that allows you to modify the data or provide fallbacks before its search record is created or updated.

1
// payload.config.ts
2
{
3
// ...
4
searchPlugin({
5
beforeSync: ({ originalDoc, searchDoc }) => ({
6
...searchDoc,
7
// - Modify your docs in any way here, this can be async
8
// - You also need to add the `excerpt` field in the `searchOverrides` config
9
excerpt: originalDoc?.excerpt || 'This is a fallback excerpt',
10
}),
11
}),
12
}

syncDrafts

When syncDrafts is true, draft documents will be synced to search. This is false by default. You must have Payload Drafts enabled for this to apply.

deleteDrafts

If true, will delete documents from search whose status changes to draft. This is true by default. You must have Payload Drafts enabled for this to apply.

TypeScript

All types can be directly imported:

1
import type { SearchConfig, BeforeSync } from '@payloadcms/plugin-search/types'
Next

Sentry Plugin