A template for Payload to power e-commerce businesses. There is a complete front-end website made explicitly for this template which can be found here.
Core features:
- Pre-configured Payload Config
- Authentication
- Access Control
- Shopping Cart
- Checkout
- Paywall
- Layout Builder
- SEO
For details on how to get this template up and running locally, see the development section.
The Payload config is tailored specifically to the needs of an e-commerce business. It is pre-configured in the following ways:
See the collections documentation for details on how to extend this functionality.
-
Users are auth-enabled and encompass both admins and customers based on the value of their
roles
field. Onlyadmin
users can access your admin panel to manage your store whereascustomer
can authenticate on your front-end to create shopping carts and place orders but have limited access to the platform, see Access Control for more details.For additional help, see the official Auth Example or the authentication docs.
-
Each product is linked to Stripe via a select field that is dynamically populated in the sidebar. This field fetches all available products in the background and displays them as options. Once a product has been selected, prices get automatically synced between Stripe and Payload. All products are layout-builder enabled so you can generate unique pages for each product using layout-building blocks, see Layout Builder for more details. Products can also gate their content or digital assets behind a paywall, see Paywall for more details.
-
When an order is placed in Stripe, a webhook is fired that Payload listens for. This webhook creates a new order in Payload with the same data as the invoice. See the Stripe section for more details.
-
All pages are layout-builder enabled so you can generate unique layouts for each page using layout-building blocks, see Layout Builder for more details.
-
This is the uploads-enabled collection used by products and pages to contain media, etc.
-
A taxonomy used to group products together. Categories can be nested inside of one another, for example "Shirts > Red". See the official Payload Nested Docs Plugin for more details.
See the globals documentation for details on how to extend this functionality.
-
Header
The data required by the header on your front-end, i.e. nav links, etc.
-
Footer
Same as above but for the footer of your site.
Basic role-based access control is setup to determine what users can and cannot do based on their roles, which are:
admin
: They can access the Payload admin panel to manage your store. They can see all data and make all operations.customer
: They cannot access the Payload admin panel and have a limited access to operations based on their user (see below).
This applies to each collection in the following ways:
users
: Only admins and the user themselves can access their profile. Anyone can create a user but only admins can delete users.orders
: Only admins and the user who placed the order can access it. Once placed, orders cannot be edited or deleted.products
: Everyone can access products, but only admins can create, update, or delete them. Paywall-enabled products may also have content that is only accessible by users who have purchased the product. See Paywall for more details.
For more details on how to extend this functionality, see the Payload Access Control docs.
Logged-in users can have their shopping carts saved to their profiles as they shop. This way they can continue shopping at a later date or on another device. When not logged in, the cart can be saved to local storage and synced to Payload on the next login. This works by maintaining a cart
field on the user
:
{
name: 'cart',
label: 'Shopping Cart',
type: 'object',
fields: [
{
name: 'items',
label: 'Items',
type: 'array',
fields: [
// product, quantity, etc
]
},
// other metadata like `createdOn`, etc
]
}
A complete front-end solution for this can be found here.
Payload itself handles no currency exchange. All payments are processed and billed using Stripe. This means you must have access to a Stripe account via an API key, see Connect Stripe for how to get one. When you create a product in Payload that wish to sell, it must be connected to a Stripe product by selecting one from the field in the products sidebar. This field fetches all available products in the background and displays them as options, see Products for more details. Once set, data is automatically synced between the two platforms in the following ways:
-
Stripe to Payload using Stripe Webhooks:
invoice.created
invoice.updated
product.created
product.updated
price.updated
-
Payload to Stripe using Payload Hooks:
user.create
For more details on how to extend this functionality, see the the official Payload Stripe Plugin.
A custom endpoint is opened at /api/checkout
which initiates the checkout process. This endpoint creates a PaymentIntent
with the items in the cart using the Stripe's Invoices API. First, an invoice is drafted, then each item in your cart is appended as a line-item to the invoice. The total price is recalculated on the server to ensure accuracy and security, and once completed, passes the client_secret
back in the response for your front-end to finalize the payment.
Products can optionally gate content or digital assets behind a paywall. This will require the product to be purchased before it's resources are accessible. To do this, we add a paywall
field to the product
collection with read
access control to check for associated purchases on each request. A purchases
field is maintained on each user to determine their access which can be manually adjusted as needed.
{
name: 'paywall',
label: 'Paywall',
type: 'blocks',
access: {
read: checkUserPurchases,
},
fields: [
// assets
]
}
Products and pages can be built using a powerful layout builder. This allows you to create unique layouts for each product or page. This template comes pre-configured with the following layout building blocks:
- Hero
- Content
- Media
- Call To Action
- Archive
A complete front-end solution for this can be found here.
This template comes pre-configured with the official Payload SEO Plugin for complete SEO control. A front-end solution for this can be found here.
To spin up the template locally, follow these steps:
- First clone the repo
- Then
cd YOUR_PROJECT_REPO && cp .env.example .env
- Next
yarn && yarn dev
(ordocker-compose up
, see Docker) - Now
open http://localhost:8000/admin
to access the admin panel - Create your first admin user using the form on the page
That's it! Changes made in ./src
will be reflected in your app—but your database is blank and your app is not yet connected to Stripe, more details on that here. You can optionally seed the database with a few products and pages, more details on that here.
To integrate with Stripe, follow these steps:
- You will first need to create a Stripe account if you do not already have one.
- Retrieve your Stripe Secret Key from the Stripe admin panel and paste it into your
env
:STRIPE_SECRET_KEY=
- In another terminal, listen for webhooks:
stripe login # follow the prompts yarn stripe:webhooks
- Paste the given webhook signing secret into your
env
:STRIPE_WEBHOOKS_ENDPOINT_SECRET=
- Reboot Payload to ensure that Stripe connects and the webhooks are registered.
See the official Payload Stripe Plugin for more details.
Alternatively, you can use Docker to spin up this template locally. To do so, follow these steps:
- Follow steps 1 and 2 from above, the docker-compose file will automatically use the
.env
file in your project root - Next run
docker-compose up
- Follow steps 4 and 5 from above to login and create your first admin user
That's it! The Docker instance will help you get up and running quickly while also standardizing the development environment across your teams.
To seed the database with a few products and pages you can run yarn seed
.
NOTICE: seeding the database is destructive because it drops your current database to populate a fresh one from the seed template. Only run this command if you are starting a new project or can afford to lose your current data.
To run Payload in production, you need to build and serve the Admin panel. To do so, follow these steps:
- First invoke the
payload build
script by runningyarn build
ornpm run build
in your project root. This creates a./build
directory with a production-ready admin bundle. - Then run
yarn serve
ornpm run serve
to run Node in production and serve Payload from the./build
directory.
The easiest way to deploy your project is to use Payload Cloud, a one-click hosting solution to deploy production-ready instances of your Payload apps directly from your GitHub repo. You can also deploy your app manually, check out the deployment documentation for full details.