Ricetta: CMS per Blog
Panoramica
Sezione intitolata “Panoramica”Costruisci un backend per blog con:
- Articoli con contenuto markdown e immagini di copertina
- Autori con profili
- Categorie con una relazione molti-a-molti
Collezioni
Sezione intitolata “Collezioni”import { EntityCollection } from "@rebasepro/types";
export const authorsCollection: EntityCollection = { slug: "authors", name: "Authors", singularName: "Author", table: "authors", icon: "person", properties: { name: { type: "string", name: "Name", validation: { required: true } }, email: { type: "string", name: "Email", email: true, validation: { required: true, unique: true } }, avatar: { type: "string", name: "Avatar", storage: { storagePath: "avatars", acceptedFiles: ["image/*"], maxSize: 2 * 1024 * 1024 } }, bio: { type: "string", name: "Bio", multiline: true } }};Categorie
Sezione intitolata “Categorie”export const categoriesCollection: EntityCollection = { slug: "categories", name: "Categories", singularName: "Category", table: "categories", icon: "label", properties: { name: { type: "string", name: "Name", validation: { required: true } }, slug: { type: "string", name: "Slug", validation: { required: true, unique: true } }, color: { type: "string", name: "Color", enum: [ { id: "blue", label: "Blue", color: "blueDark" }, { id: "green", label: "Green", color: "greenDark" }, { id: "red", label: "Red", color: "pinkDark" }, { id: "orange", label: "Orange", color: "orangeDark" } ] } }};Articoli
Sezione intitolata “Articoli”export const articlesCollection: EntityCollection = { slug: "articles", name: "Articles", singularName: "Article", table: "articles", icon: "article", defaultViewMode: "table", history: true, properties: { title: { type: "string", name: "Title", validation: { required: true } }, slug: { type: "string", name: "URL Slug", validation: { required: true, unique: true } }, author: { type: "relation", name: "Author", relationName: "author" }, status: { type: "string", name: "Status", enum: [ { id: "draft", label: "Draft", color: "grayDark" }, { id: "review", label: "In Review", color: "orangeDark" }, { id: "published", label: "Published", color: "greenDark" } ], defaultValue: "draft" }, cover_image: { type: "string", name: "Cover Image", storage: { storagePath: "articles/covers", acceptedFiles: ["image/*"] } }, content: { type: "string", name: "Content", markdown: true }, excerpt: { type: "string", name: "Excerpt", multiline: true, validation: { max: 300 } }, published_at: { type: "date", name: "Published At" }, created_at: { type: "date", name: "Created At", autoValue: "on_create", readOnly: true } }, relations: [ { relationName: "author", target: () => authorsCollection, cardinality: "one", localKey: "author_id" }, { relationName: "categories", target: () => categoriesCollection, cardinality: "many", through: { table: "article_categories", sourceColumn: "article_id", targetColumn: "category_id" } } ], callbacks: { beforeSave: async ({ values, status }) => { // Auto-generate slug if (values.title && !values.slug) { values.slug = values.title .toLowerCase() .replace(/[^a-z0-9]+/g, "-"); } // Set published_at when publishing if (values.status === "published" && !values.published_at) { values.published_at = new Date(); } return values; } }, securityRules: [ { operation: "select", access: "public", using: "{status} = 'published'" }, { operation: "select", ownerField: "author_id" }, { operations: ["insert", "update"], ownerField: "author_id" }, { operation: "delete", roles: ["admin"] } ]};Configurazione
Sezione intitolata “Configurazione”- Aggiungi tutte e tre le collezioni al tuo
config/collections/index.ts - Esegui
rebase schema generate - Esegui
rebase db push - Riavvia il server di sviluppo
Ora hai un CMS per blog completamente funzionale con:
- Gestione degli autori con caricamento di avatar
- Tagging delle categorie tramite relazioni molti-a-molti
- Editing di contenuti Markdown
- Workflow Bozza → Revisione → Pubblicato
- Slug URL generati automaticamente
- Policy RLS che limitano gli autori ai propri post
- Traccia di audit completa tramite la cronologia delle entità
Interrogazione dall’SDK
Sezione intitolata “Interrogazione dall’SDK”Usa l’SDK client per recuperare gli articoli con le loro relazioni:
// Fetch published articles with author and categories includedconst { data: articles } = await client.data.articles .where("status", "==", "published") .include("author", "categories") .orderBy("published_at", "desc") .limit(10) .find();
for (const article of articles) { console.log(article.values.title); console.log(article.values.author?.name); // Hydrated relation console.log(article.values.author_id); // Scalar FK console.log(article.values.categories); // Array of related entities}