skills/headless-modern/saleor-development/SKILL.md
Build and extend Saleor's GraphQL-based headless commerce platform with custom apps, webhook handlers, and dashboard UI customizations
npx skillsauth add finsilabs/awesome-ecommerce-skills saleor-developmentInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Saleor is a headless, GraphQL-first e-commerce platform built on Django and Python. It exposes a fully typed GraphQL API for storefronts and third-party apps, a React-based dashboard for store management, and an extension system that lets you react to events via webhooks or inject UI into the dashboard. This skill covers querying the Saleor API, building Saleor Apps (plugins hosted outside Saleor), and customizing the dashboard with App Extensions.
This skill is written for custom/headless storefronts (Node.js, Python, or similar backend). The code examples use TypeScript/Node.js and can be adapted to any stack.
Shopify: Shopify Hydrogen is Shopify's headless framework. MACH/composable patterns apply when using Shopify as the commerce backend with a custom frontend, or when mixing Shopify with other best-of-breed services. WooCommerce: WooCommerce can serve as a headless backend via its REST API and WPGraphQL. These patterns apply when decoupling the frontend from WordPress. Magento: Magento's GraphQL API and PWA Studio support headless architectures. These composable patterns apply to Magento as a backend service in a MACH stack.
You'll need:
Run Saleor locally with Docker Compose
git clone https://github.com/saleor/saleor-platform.git
cd saleor-platform
docker compose up --detach
# API: http://localhost:8000/graphql/
# Dashboard: http://localhost:9000
Create the first superuser and populate demo data:
docker compose run --rm api python manage.py createsuperuser
docker compose run --rm api python manage.py populatedb --createsuperuser
Query the Storefront GraphQL API
Use the Saleor CLI or any GraphQL client (Apollo, urql, graphql-request).
Install the CLI for code generation:
npm install -g @saleor/cli
saleor configure
Example — fetch the first 12 products from the default channel:
query ProductList($channel: String!) {
products(first: 12, channel: $channel) {
edges {
node {
id
name
slug
thumbnail { url alt }
pricing {
priceRange {
start { gross { amount currency } }
}
}
}
}
pageInfo { hasNextPage endCursor }
}
}
import { createClient } from 'urql';
const client = createClient({
url: process.env.NEXT_PUBLIC_SALEOR_API_URL,
fetchOptions: () => ({
headers: { 'Content-Type': 'application/json' },
}),
});
const { data } = await client.query(PRODUCT_LIST_QUERY, { channel: 'default-channel' }).toPromise();
Authenticate a customer and start checkout
mutation CustomerLogin($email: String!, $password: String!) {
tokenCreate(email: $email, password: $password) {
token
refreshToken
errors { field message }
user { id email }
}
}
Create a checkout and add lines:
mutation CheckoutCreate($channel: String!, $lines: [CheckoutLineInput!]!) {
checkoutCreate(input: { channel: $channel, lines: $lines }) {
checkout {
id
token
totalPrice { gross { amount currency } }
}
errors { field message }
}
}
Complete checkout with a payment gateway token (e.g., from Stripe Elements):
mutation CheckoutComplete($checkoutId: ID!, $paymentData: JSONString) {
checkoutComplete(id: $checkoutId, paymentData: $paymentData) {
order { id number status }
errors { field message code }
}
}
Bootstrap a Saleor App
A Saleor App is a Node.js service that registers itself with Saleor, receives webhooks, and optionally renders UI in the dashboard via iframes.
npx @saleor/app-sdk@latest create my-saleor-app
cd my-saleor-app
npm install
npm run dev
# Expose with: npx ngrok http 3000
Register the app in the dashboard under Apps → Install custom app, entering your ngrok URL. Saleor calls your /api/manifest endpoint:
// pages/api/manifest.ts
import { createManifestHandler } from "@saleor/app-sdk/handlers/next";
import { AppManifest } from "@saleor/app-sdk/types";
const manifest: AppManifest = {
id: "my-saleor-app",
name: "My Saleor App",
version: "1.0.0",
about: "Example app",
permissions: ["MANAGE_ORDERS"],
appUrl: process.env.APP_URL!,
tokenTargetUrl: `${process.env.APP_URL}/api/register`,
webhooks: [
{
name: "Order Created",
asyncEvents: ["ORDER_CREATED"],
query: `subscription { event { ... on OrderCreated { order { id number } } } }`,
targetUrl: `${process.env.APP_URL}/api/webhooks/order-created`,
isActive: true,
},
],
};
export default createManifestHandler({ manifestFactory: () => manifest });
Handle Saleor webhooks securely
Saleor signs every webhook with an HMAC-SHA256 signature using your app's secret token.
// pages/api/webhooks/order-created.ts
import { SaleorAsyncWebhook } from "@saleor/app-sdk/handlers/next";
import { OrderCreatedDocument } from "@/generated/graphql";
const orderCreatedWebhook = new SaleorAsyncWebhook<OrderCreatedPayload>({
name: "Order Created",
webhookPath: "api/webhooks/order-created",
asyncEvent: "ORDER_CREATED",
apl: saleorApp.apl,
query: OrderCreatedDocument,
});
export default orderCreatedWebhook.createHandler((req, res, ctx) => {
const { order } = ctx.payload;
console.log(`New order #${order.number} received`);
// Trigger fulfillment, email, ERP sync, etc.
return res.status(200).end();
});
export const config = { api: { bodyParser: false } }; // required for signature check
Add a Dashboard Extension (custom UI panel)
Extensions render an iframe inside the Saleor Dashboard. Declare them in the manifest:
extensions: [
{
label: "Sync to ERP",
mount: "PRODUCT_DETAILS_MORE_ACTIONS",
target: "POPUP",
permissions: ["MANAGE_PRODUCTS"],
url: `${process.env.APP_URL}/extension/product-sync`,
},
],
The extension page uses @saleor/app-sdk to communicate with the dashboard host:
import { actions, useAppBridge } from "@saleor/app-sdk/app-bridge";
export default function ProductSyncExtension() {
const { appBridge } = useAppBridge();
const handleSync = async () => {
appBridge?.dispatch(actions.Notification({
status: "success",
title: "Sync started",
text: "Product is being synced to ERP.",
}));
};
return <button onClick={handleSync}>Sync to ERP</button>;
}
import { GraphQLClient, gql } from 'graphql-request';
const client = new GraphQLClient(process.env.SALEOR_API_URL!, {
headers: { Authorization: `Bearer ${process.env.SALEOR_APP_TOKEN}` },
});
const PRODUCTS_QUERY = gql`
query Products($first: Int!, $after: String, $channel: String!) {
products(first: $first, after: $after, channel: $channel) {
edges { node { id name slug description } }
pageInfo { hasNextPage endCursor }
}
}
`;
async function fetchAllProducts(channel: string) {
const products = [];
let after: string | null = null;
do {
const data = await client.request(PRODUCTS_QUERY, { first: 100, after, channel });
products.push(...data.products.edges.map((e: any) => e.node));
after = data.products.pageInfo.hasNextPage ? data.products.pageInfo.endCursor : null;
} while (after);
return products;
}
mutation FulfillOrder($orderId: ID!, $input: OrderFulfillInput!) {
orderFulfill(orderId: $orderId, input: $input) {
fulfillments {
id
status
trackingNumber
}
errors { field message code }
}
}
await client.request(FULFILL_ORDER_MUTATION, {
orderId: "T3JkZXI6MTIz",
input: {
lines: [{ orderLineId: "T3JkZXJMaW5lOjQ1", stocks: [{ warehouse: "V2FyZWhvdXNlOjE=", quantity: 1 }] }],
notifyCustomer: true,
allowStockToBeExceeded: false,
},
});
saleor app generate-types or use graphql-codegen so queries are fully typedSaleorAsyncWebhook wrapper which handles HMAC verification automatically; never process unauthenticated payloadsCache-Control: s-maxage=300 on catalog API routes| Problem | Solution |
|---------|----------|
| GraphQL errors for unauthorized operations | Ensure the app has been granted the correct permissions in the manifest AND in the dashboard under App settings |
| Webhook payload is empty / fields missing | The webhook payload is defined by a GraphQL subscription query in the manifest — add the fields you need to the query property |
| tokenCreate returns null on storefront | The channel must have the storefront API enabled and an assigned country; check channel configuration in the dashboard |
| App works locally but not after deployment | The APP_URL env var must match the publicly accessible URL Saleor can reach; update the app URL in the dashboard after deployment |
| Dashboard extension iframe is blank | The extension URL must be served over HTTPS and must include Access-Control-Allow-Origin headers for the dashboard origin |
tools
Let shoppers save products to a wishlist, share it with friends, and get notified when saved items come back in stock or drop in price
development
Build a themeable storefront with design tokens and CSS custom properties that supports white-labeling, multi-brand variants, and dark mode
development
Speed up product discovery with instant search suggestions, fuzzy typo matching, and category-aware results powered by Algolia or Elasticsearch
development
Build a mobile-first storefront with thumb-friendly navigation, sticky add-to-cart buttons, and touch-optimized components for high mobile conversion