title | description | type | service | stub | i18nReady |
---|---|---|---|---|---|
Strapi y Astro |
Agrega contenido a tu proyecto de Astro utilizando el headless CMS de Strapi. |
cms |
Strapi |
false |
true |
import { Steps } from '@astrojs/starlight/components'; import { FileTree } from '@astrojs/starlight/components'; import PackageManagerTabs from '~/components/tabs/PackageManagerTabs.astro';
Strapi es un headless CMS de código abierto y personalizable.
Esta guía creará una función wrapper para conectar Strapi con Astro.
Para comenzar, necesitarás tener lo siguiente:
- Un proyecto de Astro - Si aún no tienes un proyecto de Astro, nuestra guía de instalación te ayudará a poner en marcha en poco tiempo.
- Un servidor de Strapi CMS - Puedes configurar un servidor de Strapi en un entorno local.
Para agregar la URL de Strapi a Astro, crea un archivo .env
en la raíz de tu proyecto (si aún no existe) y agrega la siguiente variable:
STRAPI_URL="http://127.0.0.1:1337" // o usa tu dirección IP
Reinicia el servidor de desarrollo para utilizar esta variable de entorno en tu proyecto Astro.
Si deseas tener IntelliSense para tu variable de entorno, puedes crear un archivo env.d.ts
en el directorio src/
y configurar ImportMetaEnv
de la siguiente manera:
interface ImportMetaEnv {
readonly STRAPI_URL: string;
}
Ahora tu directorio raíz debería incluir el/los nuevos archivo(s):
- src/ - **env.d.ts** - **.env** - astro.config.mjs - package.jsonCrea un nuevo archivo en src/lib/strapi.ts
y agrega la siguiente función envolvente para interactuar con la API de Strapi:
interface Props {
endpoint: string;
query?: Record<string, string>;
wrappedByKey?: string;
wrappedByList?: boolean;
}
/**
* Obtiene datos de la API de Strapi.
* @param endpoint - El endpoint para realizar la consulta
* @param query - Los parámetros de consulta para agregar a la URL
* @param wrappedByKey - La clave para desempaquetar la respuesta
* @param wrappedByList - Si la respuesta es una lista, desempaquétala.
* @returns
*/
export default async function fetchApi<T>({
endpoint,
query,
wrappedByKey,
wrappedByList,
}: Props): Promise<T> {
if (endpoint.startsWith('/')) {
endpoint = endpoint.slice(1);
}
const url = new URL(`${import.meta.env.STRAPI_URL}/api/${endpoint}`);
if (query) {
Object.entries(query).forEach(([key, value]) => {
url.searchParams.append(key, value);
});
}
const res = await fetch(url.toString());
let data = await res.json();
if (wrappedByKey) {
data = data[wrappedByKey];
}
if (wrappedByList) {
data = data[0];
}
return data as T;
}
Esta función requiere un objeto con las siguientes propiedades:
endpoint
- El endpoint desde el cual estás obteniendo datos.query
- Los parámetros de consulta para agregar al final de la URL.wrappedByKey
- La clavedata
en el objeto que envuelve tuResponse
.wrappedByList
- Un parámetro para "desempaquetar" la lista devuelta por Strapi y devolver solo el primer elemento.
Si estás utilizando TypeScript, crea la siguiente interfaz Article en el archivo src/interfaces/article.ts
para que coincida con los tipos de contenido de Strapi:
export default interface Article {
id: number;
attributes: {
title: string;
description: string;
content: string;
slug: string;
createdAt: string;
updatedAt: string;
publishedAt: string;
};
}
:::note Puedes modificar esta interfaz o crear múltiples interfaces para que se correspondan con los datos de tu propio proyecto. :::
- src/ - interfaces/ - **article.ts** - lib/ - strapi.ts - env.d.ts - .env - astro.config.mjs - package.json 1. Actualiza tu página de inicio `src/pages/index.astro` para mostrar una lista de publicaciones de blog, cada una con una descripción y un enlace a su propia página.-
Importa la función wrapper y la interfaz. Agrega la siguiente llamada a la API para obtener tus artículos y devolver una lista:
--- import fetchApi from '../lib/strapi'; import type Article from '../interfaces/article'; const articles = await fetchApi<Article[]>({ endpoint: 'articles', // el tipo de contenido a obtener wrappedByKey: 'data', // la clave para descomponer la respuesta }); ---
La llamada a la API solicita datos desde
http://localhost:1337/api/articles
y devuelvearticles
: un array de objetos JSON que representan tus datos:[ { id: 1, attributes: { title: "What's inside a Black Hole", description: "Maybe the answer is in this article, or not...", content: "Well, we don't know yet...", slug: "what-s-inside-a-black-hole", createdAt: "2023-05-28T13:19:46.421Z", updatedAt: "2023-05-28T13:19:46.421Z", publishedAt: "2023-05-28T13:19:45.826Z" } }, // ... ]
-
Utilizando los datos del array
articles
devuelto por la API, muestra tus publicaciones de blog de Strapi en una lista. Estas publicaciones estarán vinculadas a sus propias páginas individuales, las cuales crearás en el siguiente paso.--- import fetchApi from '../lib/strapi'; import type Article from '../interfaces/article'; const articles = await fetchApi<Article[]>({ endpoint: 'articles?populate=image', wrappedByKey: 'data', }); --- <!DOCTYPE html> <html lang="es"> <head> <title>Strapi y Astro</title> </head> <body> <main> <ul> { articles.map((article) => ( <li> <a href={`/blog/${article.attributes.slug}/`}> {article.attributes.title} </a> </li> )) } </ul> </main> </body> </html>
Crea el archivo src/pages/blog/[slug].astro
para generar dinámicamente una página para cada artículo.
En el modo estático predeterminado de Astro (SSG), utiliza getStaticPaths()
para obtener tu lista de artículos desde Strapi.
---
import fetchApi from '../../lib/strapi';
import type Article from '../../interfaces/article';
export async function getStaticPaths() {
const articles = await fetchApi<Article[]>({
endpoint: 'articles',
wrappedByKey: 'data',
});
return articles.map((article) => ({
params: { slug: article.attributes.slug },
props: article,
}));
}
type Props = Article;
const article = Astro.props;
---
Crea la plantilla para cada página utilizando las propiedades de cada objeto de publicación.
---
import fetchApi from '../../lib/strapi';
import type Article from '../../interfaces/article';
export async function getStaticPaths() {
const articles = await fetchApi<Article[]>({
endpoint: 'articles',
wrappedByKey: 'data',
});
return articles.map((article) => ({
params: { slug: article.attributes.slug },
props: article,
}));
}
type Props = Article;
const article = Astro.props;
---
<!DOCTYPE html>
<html lang="es">
<head>
<title>{article.attributes.title}</title>
</head>
<body>
<main>
<img src={import.meta.env.STRAPI_URL + article.attributes.image.data.attributes.url} />
<h1>{article.attributes.title}</h1>
<!-- Renderizar texto plano -->
<p>{article.attributes.content}</p>
<!-- Renderizar markdown -->
<MyMarkdownComponent>
{article.attributes.content}
</MyMarkdownComponent>
<!-- Renderizar html -->
<Fragment set:html={article.attributes.content} />
</main>
</body>
</html>
:::tip Asegúrate de elegir el renderizado adecuado para tu contenido. Para Markdown, consulta nuestra guía de Markdown. Si estás representando HTML, consulta esta guía para asegurar la seguridad. :::
Si has optado por el modo SSR con output: server
o output: hybrid
, genera tus rutas dinámicas utilizando el siguiente código:
Crea el archivo src/pages/blog/[slug].astro
:
---
import fetchApi from '../../../lib/strapi';
import type Article from '../../../interfaces/article';
const { slug } = Astro.params;
let article: Article;
try {
article = await fetchApi<Article>({
endpoint: 'articles',
wrappedByKey: 'data',
wrappedByList: true,
query: {
'filters[slug][$eq]': slug || '',
},
});
} catch (error) {
return Astro.redirect('/404');
}
---
<!DOCTYPE html>
<html lang="es">
<head>
<title>{article.attributes.title}</title>
</head>
<body>
<main>
<img src={import.meta.env.STRAPI_URL + article.attributes.image.data.attributes.url} />
<h1>{article.attributes.title}</h1>
<!-- Renderizar texto plano -->
<p>{article.attributes.content}</p>
<!-- Renderizar markdown -->
<MyMarkdownComponent>
{article.attributes.content}
</MyMarkdownComponent>
<!-- Renderizar html -->
<Fragment set:html={article.attributes.content} />
</main>
</body>
</html>
Este archivo obtendrá y representará los datos de la página desde Strapi que coinciden con el parámetro dinámico slug
.
Dado que estás utilizando una redirección a /404
, crea una página 404 en src/pages
:
<html lang="es">
<head>
<title>No encontrada</title>
</head>
<body>
<p>Lo siento, esta página no existe.</p>
<img src="https://http.cat/404" />
</body>
</html>
Si el artículo no se encuentra, el usuario será redirigido a esta página de error 404 y será recibido por un encantador gato.
Para desplegar tu sitio web, visita nuestras guías de implementación y sigue las instrucciones para el proveedor de hosting que prefieras.
Si tu proyecto está utilizando el modo estático predeterminado de Astro, deberás configurar un webhook para desencadenar una nueva compilación cuando tu contenido cambie. Si estás utilizando Netlify o Vercel como proveedor de hosting, puedes utilizar su función de webhook para desencadenar una nueva compilación desde Strapi.
Para configurar un webhook en Netlify:
1. Ve al panel de control de tu sitio y haz clic en **Build & deploy**.-
En la pestaña Continuous Deployment, encuentra la sección Build hooks y haz clic en Add build hook.
-
Proporciona un nombre para tu webhook y selecciona la rama en la que deseas desencadenar la compilación. Haz clic en Save y copia la URL generada.
Para configurar un webhook en Vercel:
1. Ve al panel de control de tu proyecto y haz clic en **Settings**.-
En la pestaña Git, busca la sección Deploy Hooks.
-
Proporciona un nombre para tu webhook y la rama en la que deseas desencadenar la compilación. Haz clic en Add y copia la URL generada.
Sigue la guía de webhooks de Strapi para crear un webhook en tu panel de administración de Strapi.
- Guía de Blog de Strapi para React por Strapi