Blog Template
Step-by-step guide to creating a pinned-organization blog/CMS backend with Quickback on Cloudflare Workers.
The blog template creates a fixed-org backend for a blog or CMS, deployed to Cloudflare Workers with D1. Public visitors can read content without authentication, while admins manage posts through their membership in one pinned organization.
This template uses pinned organization mode. Organization support stays on, but the generated project hard-codes one organization id and hides org switching UI.
Create the Project
quickback create blog my-site
cd my-siteThis scaffolds a project with:
my-site/
├── quickback/
│ ├── quickback.config.ts # Pinned-organization config
│ └── features/
│ └── posts/
│ ├── posts.ts # Schema + security (defineTable)
│ └── actions.ts # Publish/unpublish actions
├── src/ # Compiled output (generated)
├── package.json
├── tsconfig.json
├── wrangler.toml
└── drizzle.config.tsGenerated Configuration
quickback.config.ts
export default {
name: "my-site",
template: "hono",
features: {
pinnedOrganizationId: "org_replace_me",
},
providers: {
runtime: { name: "cloudflare", config: {} },
database: {
name: "cloudflare-d1",
config: { binding: "DB" },
},
auth: { name: "better-auth", config: {} },
},
};Replace org_replace_me with a real Better Auth organization id before deploying. Unlike the removed single-tenant mode, this template keeps the Better Auth organization plugin and resolves roles from membership in that pinned org. See Pinned Organization Mode for details.
Posts Feature (posts.ts)
The template includes a posts feature with public read access and admin-only write access:
import { sqliteTable, text, integer } from "drizzle-orm/sqlite-core";
import { defineTable } from "@quickback/compiler";
export const posts = sqliteTable("posts", {
id: text("id").primaryKey(),
title: text("title").notNull(),
slug: text("slug").notNull().unique(),
content: text("content"),
excerpt: text("excerpt"),
status: text("status").notNull().default("draft"),
publishedAt: text("published_at"),
ownerId: text("owner_id").notNull(),
});
export default defineTable(posts, {
firewall: [
{ field: 'ownerId', equals: 'ctx.userId' },
{ field: 'deletedAt', isNull: true },
],
guards: {
createable: ["title", "slug", "content", "excerpt", "status", "publishedAt"],
updatable: ["title", "slug", "content", "excerpt", "status", "publishedAt"],
immutable: ["id", "ownerId"],
},
read: {
access: { roles: ["PUBLIC"] },
},
create: { access: { roles: ["admin"] } },
update: { access: { roles: ["admin"] } },
delete: { access: { roles: ["admin"] }, mode: "soft" },
});PUBLICrole onread.accessmeans no authentication required forGET /andGET /:idadminrole oncreate,update, anddeleterequires an authenticated user withmember.role === "admin"in the pinned organizationownerfirewall scope tracks who created each post viaownerIdsoftDeletemarks posts as deleted rather than removing them from the database
Post Actions (actions.ts)
The template includes publish/unpublish actions with guard conditions:
import { posts } from './posts';
import { defineActions } from '@quickback/compiler';
import { z } from 'zod';
export default defineActions(posts, {
publish: {
description: "Publish a draft post",
input: z.object({}),
guard: {
roles: ["admin"],
record: { status: { equals: "draft" } },
},
sideEffects: "sync",
},
unpublish: {
description: "Revert a published post to draft",
input: z.object({}),
guard: {
roles: ["admin"],
record: { status: { equals: "published" } },
},
sideEffects: "sync",
},
});These actions are admin-only and include record-level guards — you can only publish drafts and unpublish published posts.
Setup Steps
1. Install Dependencies
npm install2. Log In to the Compiler
quickback login3. Compile Your Definitions
quickback compile4. Create D1 Database
The blog template uses a single D1 database (no split databases):
npx wrangler d1 create my-siteCopy the database ID from the output and update your wrangler.toml:
[[d1_databases]]
binding = "DB"
database_name = "my-site"
database_id = "paste-id-here"5. Run Migrations (Local)
npm run db:migrate:local6. Start Development Server
npm run devYour API is running at http://localhost:8787.
Using the API
Public Endpoints (No Auth Required)
# List all posts
curl http://localhost:8787/api/v1/posts
# Get a single post
curl http://localhost:8787/api/v1/posts/post-id-hereAdmin Endpoints (Auth Required)
# Create a post (admin only)
curl -X POST http://localhost:8787/api/v1/posts \
-H "Authorization: Bearer $TOKEN" \
-H "Content-Type: application/json" \
-d '{"title": "Hello World", "slug": "hello-world", "content": "..."}'
# Publish a post (admin only)
curl -X POST http://localhost:8787/api/v1/posts/post-id/actions/publish \
-H "Authorization: Bearer $TOKEN"Pairing with a Frontend
Since this is a JSON API, pair it with any frontend framework:
// Astro, Next.js, SvelteKit, etc.
const posts = await fetch('https://api.mysite.com/api/v1/posts').then(r => r.json());
// Admin operations require authentication
const res = await fetch('https://api.mysite.com/api/v1/posts', {
method: 'POST',
headers: {
'Authorization': `Bearer ${token}`,
'Content-Type': 'application/json',
},
body: JSON.stringify({ title: 'New Post', slug: 'new-post', content: '...' }),
});Adding More Features
Add new content types by creating feature directories. For example, a pages feature for static pages:
// quickback/features/pages/pages.ts
import { sqliteTable, text } from "drizzle-orm/sqlite-core";
import { defineTable } from "@quickback/compiler";
export const pages = sqliteTable("pages", {
id: text("id").primaryKey(),
title: text("title").notNull(),
slug: text("slug").notNull().unique(),
content: text("content"),
ownerId: text("owner_id").notNull(),
});
export default defineTable(pages, {
firewall: [{ field: 'ownerId', equals: 'ctx.userId' }],
read: {
access: { roles: ["PUBLIC"] },
},
create: { access: { roles: ["admin"] } },
update: { access: { roles: ["admin"] } },
delete: { access: { roles: ["admin"] } },
});Then recompile:
quickback compileDeploying to Production
# Apply remote migrations
npm run db:migrate:remote
# Deploy to Cloudflare
npm run deployNext Steps
- Pinned Organization Mode — How fixed-org mode works
- Access Control — PUBLIC role and role-based permissions
- Custom Actions — Add business logic beyond CRUD
- Cloudflare Template — Multi-tenant alternative